2011-09-28 09:16:26 -04:00
|
|
|
|
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
2010-01-13 15:05:20 -05:00
|
|
|
|
|
2011-01-13 15:04:37 -05:00
|
|
|
|
const Clutter = imports.gi.Clutter;
|
2010-01-13 15:05:20 -05:00
|
|
|
|
const DBus = imports.dbus;
|
|
|
|
|
const GLib = imports.gi.GLib;
|
|
|
|
|
const Lang = imports.lang;
|
|
|
|
|
const Shell = imports.gi.Shell;
|
|
|
|
|
const Mainloop = imports.mainloop;
|
2010-03-08 08:13:59 -05:00
|
|
|
|
const St = imports.gi.St;
|
2010-01-13 15:05:20 -05:00
|
|
|
|
|
2011-01-11 16:34:09 -05:00
|
|
|
|
const Config = imports.misc.config;
|
2010-01-13 15:05:20 -05:00
|
|
|
|
const Main = imports.ui.main;
|
|
|
|
|
const MessageTray = imports.ui.messageTray;
|
|
|
|
|
const Params = imports.misc.params;
|
2010-12-17 15:30:12 -05:00
|
|
|
|
const Util = imports.misc.util;
|
2010-01-13 15:05:20 -05:00
|
|
|
|
|
|
|
|
|
let nextNotificationId = 1;
|
|
|
|
|
|
2010-02-15 12:27:28 -05:00
|
|
|
|
// Should really be defined in dbus.js
|
|
|
|
|
const BusIface = {
|
|
|
|
|
name: 'org.freedesktop.DBus',
|
|
|
|
|
methods: [{ name: 'GetConnectionUnixProcessID',
|
|
|
|
|
inSignature: 's',
|
|
|
|
|
outSignature: 'i' }]
|
2010-03-15 09:50:05 -04:00
|
|
|
|
};
|
2010-02-15 12:27:28 -05:00
|
|
|
|
|
|
|
|
|
const Bus = function () {
|
|
|
|
|
this._init();
|
2010-03-15 09:50:05 -04:00
|
|
|
|
};
|
2010-02-15 12:27:28 -05:00
|
|
|
|
|
|
|
|
|
Bus.prototype = {
|
|
|
|
|
_init: function() {
|
|
|
|
|
DBus.session.proxifyObject(this, 'org.freedesktop.DBus', '/org/freedesktop/DBus');
|
|
|
|
|
}
|
2010-03-15 09:50:05 -04:00
|
|
|
|
};
|
2010-02-15 12:27:28 -05:00
|
|
|
|
|
|
|
|
|
DBus.proxifyPrototype(Bus.prototype, BusIface);
|
|
|
|
|
|
2010-01-13 15:05:20 -05:00
|
|
|
|
const NotificationDaemonIface = {
|
|
|
|
|
name: 'org.freedesktop.Notifications',
|
|
|
|
|
methods: [{ name: 'Notify',
|
|
|
|
|
inSignature: 'susssasa{sv}i',
|
|
|
|
|
outSignature: 'u'
|
|
|
|
|
},
|
|
|
|
|
{ name: 'CloseNotification',
|
|
|
|
|
inSignature: 'u',
|
|
|
|
|
outSignature: ''
|
|
|
|
|
},
|
|
|
|
|
{ name: 'GetCapabilities',
|
|
|
|
|
inSignature: '',
|
|
|
|
|
outSignature: 'as'
|
|
|
|
|
},
|
|
|
|
|
{ name: 'GetServerInformation',
|
|
|
|
|
inSignature: '',
|
|
|
|
|
outSignature: 'ssss'
|
|
|
|
|
}],
|
|
|
|
|
signals: [{ name: 'NotificationClosed',
|
|
|
|
|
inSignature: 'uu' },
|
|
|
|
|
{ name: 'ActionInvoked',
|
|
|
|
|
inSignature: 'us' }]
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const NotificationClosedReason = {
|
|
|
|
|
EXPIRED: 1,
|
|
|
|
|
DISMISSED: 2,
|
|
|
|
|
APP_CLOSED: 3,
|
|
|
|
|
UNDEFINED: 4
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const Urgency = {
|
|
|
|
|
LOW: 0,
|
|
|
|
|
NORMAL: 1,
|
|
|
|
|
CRITICAL: 2
|
|
|
|
|
};
|
|
|
|
|
|
2010-02-16 17:32:19 -05:00
|
|
|
|
const rewriteRules = {
|
|
|
|
|
'XChat': [
|
|
|
|
|
{ pattern: /^XChat: Private message from: (\S*) \(.*\)$/,
|
2010-11-29 11:12:21 -05:00
|
|
|
|
replacement: '<$1>' },
|
2010-02-16 17:32:19 -05:00
|
|
|
|
{ pattern: /^XChat: New public message from: (\S*) \((.*)\)$/,
|
2010-11-29 11:12:21 -05:00
|
|
|
|
replacement: '$2 <$1>' },
|
2010-02-16 17:32:19 -05:00
|
|
|
|
{ pattern: /^XChat: Highlighted message from: (\S*) \((.*)\)$/,
|
2010-11-29 11:12:21 -05:00
|
|
|
|
replacement: '$2 <$1>' }
|
2010-02-16 17:32:19 -05:00
|
|
|
|
]
|
|
|
|
|
};
|
2010-06-23 15:20:39 -04:00
|
|
|
|
|
2010-01-13 15:05:20 -05:00
|
|
|
|
function NotificationDaemon() {
|
|
|
|
|
this._init();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NotificationDaemon.prototype = {
|
|
|
|
|
_init: function() {
|
|
|
|
|
DBus.session.exportObject('/org/freedesktop/Notifications', this);
|
|
|
|
|
|
2010-08-09 13:18:15 -04:00
|
|
|
|
this._sources = {};
|
2010-09-02 10:45:53 -04:00
|
|
|
|
this._senderToPid = {};
|
|
|
|
|
this._notifications = {};
|
|
|
|
|
this._busProxy = new Bus();
|
2010-02-21 00:25:23 -05:00
|
|
|
|
|
2010-08-09 12:33:34 -04:00
|
|
|
|
Main.statusIconDispatcher.connect('message-icon-added', Lang.bind(this, this._onTrayIconAdded));
|
|
|
|
|
Main.statusIconDispatcher.connect('message-icon-removed', Lang.bind(this, this._onTrayIconRemoved));
|
|
|
|
|
|
2010-02-21 00:25:23 -05:00
|
|
|
|
Shell.WindowTracker.get_default().connect('notify::focus-app',
|
|
|
|
|
Lang.bind(this, this._onFocusAppChanged));
|
2010-02-24 15:46:00 -05:00
|
|
|
|
Main.overview.connect('hidden',
|
|
|
|
|
Lang.bind(this, this._onFocusAppChanged));
|
2010-01-13 15:05:20 -05:00
|
|
|
|
},
|
|
|
|
|
|
2010-08-05 13:09:27 -04:00
|
|
|
|
_iconForNotificationData: function(icon, hints, size) {
|
|
|
|
|
let textureCache = St.TextureCache.get_default();
|
|
|
|
|
|
2011-10-17 17:50:02 -04:00
|
|
|
|
// 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.
|
2010-08-05 13:09:27 -04:00
|
|
|
|
if (icon) {
|
|
|
|
|
if (icon.substr(0, 7) == 'file://')
|
|
|
|
|
return textureCache.load_uri_async(icon, size, size);
|
|
|
|
|
else if (icon[0] == '/') {
|
|
|
|
|
let uri = GLib.filename_to_uri(icon, null);
|
|
|
|
|
return textureCache.load_uri_async(uri, size, size);
|
|
|
|
|
} else
|
2010-11-02 18:33:22 -04:00
|
|
|
|
return new St.Icon({ icon_name: icon,
|
|
|
|
|
icon_type: St.IconType.FULLCOLOR,
|
|
|
|
|
icon_size: size });
|
2011-10-17 17:50:02 -04:00
|
|
|
|
} else if (hints['image-data']) {
|
|
|
|
|
let [width, height, rowStride, hasAlpha,
|
|
|
|
|
bitsPerSample, nChannels, data] = hints['image-data'];
|
|
|
|
|
return textureCache.load_from_raw(data, hasAlpha, width, height, rowStride, size);
|
|
|
|
|
} else if (hints['image-path']) {
|
|
|
|
|
return textureCache.load_uri_async(GLib.filename_to_uri(hints['image-path'], null), size, size);
|
2010-08-05 13:09:27 -04:00
|
|
|
|
} else {
|
|
|
|
|
let stockIcon;
|
|
|
|
|
switch (hints.urgency) {
|
|
|
|
|
case Urgency.LOW:
|
|
|
|
|
case Urgency.NORMAL:
|
|
|
|
|
stockIcon = 'gtk-dialog-info';
|
|
|
|
|
break;
|
|
|
|
|
case Urgency.CRITICAL:
|
|
|
|
|
stockIcon = 'gtk-dialog-error';
|
|
|
|
|
break;
|
|
|
|
|
}
|
2010-11-02 18:33:22 -04:00
|
|
|
|
return new St.Icon({ icon_name: stockIcon,
|
|
|
|
|
icon_type: St.IconType.FULLCOLOR,
|
|
|
|
|
icon_size: size });
|
2010-08-05 13:09:27 -04:00
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
2010-12-15 16:30:50 -05:00
|
|
|
|
// Returns the source associated with ndata.notification if it is set.
|
|
|
|
|
// Otherwise, returns the source associated with the pid if one is
|
|
|
|
|
// stored in this._sources and the notification is not transient.
|
|
|
|
|
// Otherwise, creates a new source as long as pid is provided.
|
|
|
|
|
//
|
|
|
|
|
// Either a pid or ndata.notification is needed to retrieve or
|
|
|
|
|
// create a source.
|
2011-07-06 17:01:07 -04:00
|
|
|
|
_getSource: function(title, pid, ndata, sender) {
|
2010-12-15 16:30:50 -05:00
|
|
|
|
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 isForTransientNotification = (ndata && ndata.hints['transient'] == true);
|
|
|
|
|
|
|
|
|
|
// We don't want to override a persistent notification
|
|
|
|
|
// with a transient one from the same sender, so we
|
|
|
|
|
// always create a new source object for new transient notifications
|
|
|
|
|
// and never add it to this._sources .
|
2011-06-08 02:27:57 -04:00
|
|
|
|
if (!isForTransientNotification && this._sources[pid]) {
|
|
|
|
|
let source = this._sources[pid];
|
|
|
|
|
source.setTitle(title);
|
|
|
|
|
return source;
|
|
|
|
|
}
|
2010-12-15 16:30:50 -05:00
|
|
|
|
|
2011-07-06 17:01:07 -04:00
|
|
|
|
let source = new Source(title, pid, sender);
|
2010-12-15 16:30:50 -05:00
|
|
|
|
source.setTransient(isForTransientNotification);
|
2010-08-09 12:33:34 -04:00
|
|
|
|
|
2010-12-15 16:30:50 -05:00
|
|
|
|
if (!isForTransientNotification) {
|
|
|
|
|
this._sources[pid] = source;
|
|
|
|
|
source.connect('destroy', Lang.bind(this,
|
|
|
|
|
function() {
|
|
|
|
|
delete this._sources[pid];
|
|
|
|
|
}));
|
|
|
|
|
}
|
2010-08-09 12:33:34 -04:00
|
|
|
|
|
|
|
|
|
Main.messageTray.add(source);
|
|
|
|
|
return source;
|
|
|
|
|
},
|
|
|
|
|
|
2010-01-13 15:05:20 -05:00
|
|
|
|
Notify: function(appName, replacesId, icon, summary, body,
|
|
|
|
|
actions, hints, timeout) {
|
2010-09-02 10:45:53 -04:00
|
|
|
|
let id;
|
2010-01-13 15:05:20 -05:00
|
|
|
|
|
2011-07-11 07:09:45 -04:00
|
|
|
|
// Filter out chat, presence, calls and invitation notifications from
|
|
|
|
|
// Empathy, since we handle that information from telepathyClient.js
|
2011-03-29 04:04:47 -04:00
|
|
|
|
if (appName == 'Empathy' && (hints['category'] == 'im.received' ||
|
2011-06-30 08:05:41 -04:00
|
|
|
|
hints['category'] == 'x-empathy.im.room-invitation' ||
|
2011-07-11 07:09:45 -04:00
|
|
|
|
hints['category'] == 'x-empathy.call.incoming' ||
|
2011-08-16 11:07:40 -04:00
|
|
|
|
hints['category'] == 'x-empathy.call.incoming"' ||
|
2011-08-26 06:27:40 -04:00
|
|
|
|
hints['category'] == 'x-empathy.im.subscription-request' ||
|
2011-03-29 04:04:47 -04:00
|
|
|
|
hints['category'] == 'presence.online' ||
|
|
|
|
|
hints['category'] == 'presence.offline')) {
|
2010-09-02 10:45:53 -04:00
|
|
|
|
// Ignore replacesId since we already sent back a
|
|
|
|
|
// NotificationClosed for that id.
|
2010-02-02 10:21:47 -05:00
|
|
|
|
id = nextNotificationId++;
|
|
|
|
|
Mainloop.idle_add(Lang.bind(this,
|
|
|
|
|
function () {
|
|
|
|
|
this._emitNotificationClosed(id, NotificationClosedReason.DISMISSED);
|
|
|
|
|
}));
|
|
|
|
|
return id;
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-16 17:32:19 -05:00
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2010-09-02 10:45:53 -04:00
|
|
|
|
hints = Params.parse(hints, { urgency: Urgency.NORMAL }, true);
|
|
|
|
|
|
2011-08-29 13:11:24 -04:00
|
|
|
|
// 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'])
|
2011-01-19 10:30:01 -05:00
|
|
|
|
if (hints['image_data'])
|
|
|
|
|
hints['image-data'] = hints['image_data']; // version 1.1 of the spec
|
2011-08-29 13:11:24 -04:00
|
|
|
|
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'];
|
2011-01-19 10:30:01 -05:00
|
|
|
|
|
2010-09-02 10:45:53 -04:00
|
|
|
|
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 = nextNotificationId++;
|
|
|
|
|
}
|
|
|
|
|
this._notifications[id] = ndata;
|
|
|
|
|
|
|
|
|
|
let sender = DBus.getCurrentMessageContext().sender;
|
|
|
|
|
let pid = this._senderToPid[sender];
|
2010-12-15 16:30:50 -05:00
|
|
|
|
|
2011-07-06 17:01:07 -04:00
|
|
|
|
let source = this._getSource(appName, pid, ndata, sender);
|
2010-09-02 10:45:53 -04:00
|
|
|
|
|
|
|
|
|
if (source) {
|
|
|
|
|
this._notifyForSource(source, ndata);
|
|
|
|
|
return id;
|
2010-02-23 10:50:35 -05:00
|
|
|
|
}
|
|
|
|
|
|
2010-09-02 10:45:53 -04:00
|
|
|
|
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 id;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this._busProxy.GetConnectionUnixProcessIDRemote(sender, Lang.bind(this,
|
|
|
|
|
function (pid, ex) {
|
|
|
|
|
// The app may have updated or removed the notification
|
|
|
|
|
ndata = this._notifications[id];
|
|
|
|
|
if (!ndata)
|
|
|
|
|
return;
|
|
|
|
|
|
2011-07-06 17:01:07 -04:00
|
|
|
|
source = this._getSource(appName, pid, ndata, sender);
|
2010-12-15 16:30:50 -05:00
|
|
|
|
|
|
|
|
|
// We only store sender-pid entries for persistent sources.
|
|
|
|
|
// Removing the entries once the source is destroyed
|
|
|
|
|
// would result in the entries associated with transient
|
|
|
|
|
// sources removed once the notification is shown anyway.
|
|
|
|
|
// However, keeping these pairs would mean that we would
|
|
|
|
|
// possibly remove an entry associated with a persistent
|
|
|
|
|
// source when a transient source for the same sender is
|
|
|
|
|
// distroyed.
|
|
|
|
|
if (!source.isTransient) {
|
|
|
|
|
this._senderToPid[sender] = pid;
|
|
|
|
|
source.connect('destroy', Lang.bind(this,
|
|
|
|
|
function() {
|
|
|
|
|
delete this._senderToPid[sender];
|
|
|
|
|
}));
|
|
|
|
|
}
|
2010-09-02 10:45:53 -04:00
|
|
|
|
this._notifyForSource(source, ndata);
|
|
|
|
|
}));
|
|
|
|
|
|
|
|
|
|
return id;
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
_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];
|
|
|
|
|
|
2010-08-05 13:09:27 -04:00
|
|
|
|
let iconActor = this._iconForNotificationData(icon, hints, source.ICON_SIZE);
|
2010-09-02 10:45:53 -04:00
|
|
|
|
|
2010-02-22 17:19:32 -05:00
|
|
|
|
if (notification == null) {
|
2010-11-23 18:31:55 -05:00
|
|
|
|
notification = new MessageTray.Notification(source, summary, body,
|
|
|
|
|
{ icon: iconActor,
|
|
|
|
|
bannerMarkup: true });
|
2010-09-02 10:45:53 -04:00
|
|
|
|
ndata.notification = notification;
|
2010-07-14 17:07:06 -04:00
|
|
|
|
notification.connect('destroy', Lang.bind(this,
|
2011-01-27 18:26:53 -05:00
|
|
|
|
function(n, reason) {
|
2011-08-25 09:20:00 -04:00
|
|
|
|
delete this._notifications[ndata.id];
|
2011-01-27 18:26:53 -05:00
|
|
|
|
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;
|
|
|
|
|
}
|
2011-08-25 09:20:00 -04:00
|
|
|
|
this._emitNotificationClosed(ndata.id, notificationClosedReason);
|
2010-07-14 17:07:06 -04:00
|
|
|
|
}));
|
2010-12-16 15:49:47 -05:00
|
|
|
|
notification.connect('action-invoked', Lang.bind(this,
|
|
|
|
|
function(n, actionId) {
|
2011-08-25 09:20:00 -04:00
|
|
|
|
this._emitActionInvoked(ndata.id, actionId);
|
2010-12-16 15:49:47 -05:00
|
|
|
|
}));
|
2010-02-22 17:19:32 -05:00
|
|
|
|
} else {
|
2010-08-05 13:09:27 -04:00
|
|
|
|
notification.update(summary, body, { icon: iconActor,
|
2010-11-23 18:31:55 -05:00
|
|
|
|
bannerMarkup: true,
|
2010-08-05 13:09:27 -04:00
|
|
|
|
clear: true });
|
2010-02-22 17:19:32 -05:00
|
|
|
|
}
|
2011-08-29 13:11:24 -04:00
|
|
|
|
|
2011-10-17 17:50:02 -04:00
|
|
|
|
// We only display a large image if an icon is also specified.
|
|
|
|
|
if (icon && (hints['image-data'] || hints['image-path'])) {
|
2011-08-29 13:11:24 -04:00
|
|
|
|
let image = null;
|
|
|
|
|
if (hints['image-data']) {
|
|
|
|
|
let [width, height, rowStride, hasAlpha,
|
|
|
|
|
bitsPerSample, nChannels, data] = hints['image-data'];
|
|
|
|
|
image = St.TextureCache.get_default().load_from_raw(data, hasAlpha,
|
|
|
|
|
width, height, rowStride, notification.IMAGE_SIZE);
|
|
|
|
|
} else if (hints['image-path']) {
|
|
|
|
|
image = St.TextureCache.get_default().load_uri_async(GLib.filename_to_uri(hints['image-path'], null),
|
|
|
|
|
notification.IMAGE_SIZE,
|
|
|
|
|
notification.IMAGE_SIZE);
|
|
|
|
|
}
|
|
|
|
|
notification.setImage(image);
|
|
|
|
|
} else {
|
|
|
|
|
notification.unsetImage();
|
|
|
|
|
}
|
2010-02-22 17:19:32 -05:00
|
|
|
|
|
2010-02-01 15:41:22 -05:00
|
|
|
|
if (actions.length) {
|
2010-10-30 02:29:21 -04:00
|
|
|
|
notification.setUseActionIcons(hints['action-icons'] == true);
|
2011-08-02 09:42:45 -04:00
|
|
|
|
for (let i = 0; i < actions.length - 1; i += 2) {
|
|
|
|
|
if (actions[i] == 'default')
|
|
|
|
|
notification.connect('clicked', Lang.bind(this,
|
|
|
|
|
function() {
|
2011-08-25 09:20:00 -04:00
|
|
|
|
this._emitActionInvoked(ndata.id, "default");
|
2011-08-03 15:21:00 -04:00
|
|
|
|
}));
|
2011-08-02 09:42:45 -04:00
|
|
|
|
else
|
|
|
|
|
notification.addButton(actions[i], actions[i + 1]);
|
|
|
|
|
}
|
2010-02-01 15:41:22 -05:00
|
|
|
|
}
|
2011-01-04 04:34:57 -05:00
|
|
|
|
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;
|
|
|
|
|
}
|
2010-12-16 15:49:47 -05:00
|
|
|
|
notification.setResident(hints.resident == true);
|
2010-12-15 16:30:50 -05:00
|
|
|
|
// '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);
|
2010-04-28 15:34:27 -04:00
|
|
|
|
|
2010-08-09 12:33:34 -04:00
|
|
|
|
let sourceIconActor = source.useNotificationIcon ? this._iconForNotificationData(icon, hints, source.ICON_SIZE) : null;
|
2011-03-21 17:26:04 -04:00
|
|
|
|
source.processNotification(notification, sourceIconActor);
|
2010-01-13 15:05:20 -05:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
CloseNotification: function(id) {
|
2010-09-02 10:45:53 -04:00
|
|
|
|
let ndata = this._notifications[id];
|
|
|
|
|
if (ndata) {
|
|
|
|
|
if (ndata.notification)
|
2011-01-27 18:26:53 -05:00
|
|
|
|
ndata.notification.destroy(MessageTray.NotificationDestroyedReason.SOURCE_CLOSED);
|
2010-09-02 10:45:53 -04:00
|
|
|
|
delete this._notifications[id];
|
|
|
|
|
}
|
2010-01-13 15:05:20 -05:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
GetCapabilities: function() {
|
|
|
|
|
return [
|
2010-02-01 15:41:22 -05:00
|
|
|
|
'actions',
|
2010-10-30 02:29:21 -04:00
|
|
|
|
'action-icons',
|
2010-01-13 15:05:20 -05:00
|
|
|
|
'body',
|
|
|
|
|
// 'body-hyperlinks',
|
|
|
|
|
// 'body-images',
|
|
|
|
|
'body-markup',
|
|
|
|
|
// 'icon-multi',
|
2010-06-11 17:19:18 -04:00
|
|
|
|
'icon-static',
|
2010-10-08 22:34:04 -04:00
|
|
|
|
'persistence',
|
2010-01-13 15:05:20 -05:00
|
|
|
|
// 'sound',
|
|
|
|
|
];
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
GetServerInformation: function() {
|
|
|
|
|
return [
|
2011-01-11 16:34:09 -05:00
|
|
|
|
Config.PACKAGE_NAME,
|
2010-01-13 15:05:20 -05:00
|
|
|
|
'GNOME',
|
2011-01-11 16:34:09 -05:00
|
|
|
|
Config.PACKAGE_VERSION,
|
|
|
|
|
'1.2'
|
2010-01-13 15:05:20 -05:00
|
|
|
|
];
|
|
|
|
|
},
|
|
|
|
|
|
2010-02-24 15:46:00 -05:00
|
|
|
|
_onFocusAppChanged: function() {
|
|
|
|
|
let tracker = Shell.WindowTracker.get_default();
|
2010-08-09 13:18:15 -04:00
|
|
|
|
if (!tracker.focus_app)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
for (let id in this._sources) {
|
|
|
|
|
let source = this._sources[id];
|
|
|
|
|
if (source.app == tracker.focus_app) {
|
2011-03-21 17:43:34 -04:00
|
|
|
|
source.destroyNonResidentNotifications();
|
2010-08-09 13:18:15 -04:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
2010-02-21 00:25:23 -05:00
|
|
|
|
},
|
|
|
|
|
|
2010-01-13 15:05:20 -05:00
|
|
|
|
_emitNotificationClosed: function(id, reason) {
|
|
|
|
|
DBus.session.emit_signal('/org/freedesktop/Notifications',
|
|
|
|
|
'org.freedesktop.Notifications',
|
|
|
|
|
'NotificationClosed', 'uu',
|
|
|
|
|
[id, reason]);
|
2010-02-01 15:41:22 -05:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
_emitActionInvoked: function(id, action) {
|
|
|
|
|
DBus.session.emit_signal('/org/freedesktop/Notifications',
|
|
|
|
|
'org.freedesktop.Notifications',
|
|
|
|
|
'ActionInvoked', 'us',
|
|
|
|
|
[id, action]);
|
2010-08-09 12:33:34 -04:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
_onTrayIconAdded: function(o, icon) {
|
2011-07-06 17:01:07 -04:00
|
|
|
|
let source = this._getSource(icon.title || icon.wm_class || _("Unknown"), icon.pid, null, null);
|
2010-08-09 12:33:34 -04:00
|
|
|
|
source.setTrayIcon(icon);
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
_onTrayIconRemoved: function(o, icon) {
|
|
|
|
|
let source = this._sources[icon.pid];
|
|
|
|
|
if (source)
|
|
|
|
|
source.destroy();
|
2010-01-13 15:05:20 -05:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
DBus.conformExport(NotificationDaemon.prototype, NotificationDaemonIface);
|
|
|
|
|
|
2011-07-06 17:01:07 -04:00
|
|
|
|
function Source(title, pid, sender) {
|
|
|
|
|
this._init(title, pid, sender);
|
2010-01-13 15:05:20 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Source.prototype = {
|
|
|
|
|
__proto__: MessageTray.Source.prototype,
|
|
|
|
|
|
2011-07-06 17:01:07 -04:00
|
|
|
|
_init: function(title, pid, sender) {
|
2010-08-09 13:18:15 -04:00
|
|
|
|
MessageTray.Source.prototype._init.call(this, title);
|
2010-01-13 15:05:20 -05:00
|
|
|
|
|
2010-08-09 12:33:34 -04:00
|
|
|
|
this._pid = pid;
|
2011-07-06 17:01:07 -04:00
|
|
|
|
if (sender)
|
|
|
|
|
// TODO: dbus-glib implementation of watch_name() doesn’t return an id to be used for
|
|
|
|
|
// unwatch_name() or implement unwatch_name(), however when we move to using GDBus implementation,
|
|
|
|
|
// we should save the id here and call unwatch_name() with it in destroy().
|
|
|
|
|
// Moving to GDBus is the work in progress: https://bugzilla.gnome.org/show_bug.cgi?id=648651
|
|
|
|
|
// and https://bugzilla.gnome.org/show_bug.cgi?id=622921 .
|
|
|
|
|
DBus.session.watch_name(sender,
|
|
|
|
|
false,
|
|
|
|
|
null,
|
|
|
|
|
Lang.bind(this, this._onNameVanished));
|
|
|
|
|
|
2010-08-09 12:33:34 -04:00
|
|
|
|
this._setApp();
|
|
|
|
|
if (this.app)
|
2010-09-02 10:45:53 -04:00
|
|
|
|
this.title = this.app.get_name();
|
2010-08-09 12:33:34 -04:00
|
|
|
|
else
|
|
|
|
|
this.useNotificationIcon = true;
|
2011-01-13 15:04:37 -05:00
|
|
|
|
this._trayIcon = null;
|
2010-01-13 15:05:20 -05:00
|
|
|
|
},
|
|
|
|
|
|
2011-07-06 17:01:07 -04:00
|
|
|
|
_onNameVanished: function() {
|
|
|
|
|
// Destroy the notification source when its sender is removed from DBus.
|
2011-07-27 17:08:08 -04:00
|
|
|
|
// Only do so if this.app is set to avoid removing "notify-send" sources, senders
|
|
|
|
|
// of which аre removed from DBus immediately.
|
2011-07-06 17:01:07 -04:00
|
|
|
|
// 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.
|
2011-07-27 17:08:08 -04:00
|
|
|
|
if (!this.trayIcon && this.app)
|
2011-07-06 17:01:07 -04:00
|
|
|
|
this.destroy();
|
|
|
|
|
},
|
|
|
|
|
|
2011-03-21 17:26:04 -04:00
|
|
|
|
processNotification: function(notification, icon) {
|
2010-08-09 12:33:34 -04:00
|
|
|
|
if (!this.app)
|
|
|
|
|
this._setApp();
|
|
|
|
|
if (!this.app && icon)
|
2010-09-02 10:45:53 -04:00
|
|
|
|
this._setSummaryIcon(icon);
|
2011-03-21 17:26:04 -04:00
|
|
|
|
|
|
|
|
|
let tracker = Shell.WindowTracker.get_default();
|
|
|
|
|
if (notification.resident && this.app && tracker.focus_app == this.app)
|
|
|
|
|
this.pushNotification(notification);
|
|
|
|
|
else
|
|
|
|
|
this.notify(notification);
|
2010-02-15 12:27:28 -05:00
|
|
|
|
},
|
|
|
|
|
|
2011-01-13 15:04:37 -05:00
|
|
|
|
handleSummaryClick: function() {
|
|
|
|
|
if (!this._trayIcon)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
let event = Clutter.get_current_event();
|
|
|
|
|
if (event.type() != Clutter.EventType.BUTTON_RELEASE)
|
|
|
|
|
return false;
|
|
|
|
|
|
2011-03-26 10:57:23 -04:00
|
|
|
|
// 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 (event.get_button() == 1 &&
|
|
|
|
|
this.notifications.length > 0)
|
|
|
|
|
return false;
|
|
|
|
|
|
2011-03-16 13:16:38 -04:00
|
|
|
|
if (Main.overview.visible) {
|
|
|
|
|
// We can't just connect to Main.overview's 'hidden' signal,
|
|
|
|
|
// because it's emitted *before* it calls popModal()...
|
|
|
|
|
let id = global.connect('notify::stage-input-mode', Lang.bind(this,
|
|
|
|
|
function () {
|
|
|
|
|
global.disconnect(id);
|
|
|
|
|
this._trayIcon.click(event);
|
|
|
|
|
}));
|
|
|
|
|
Main.overview.hide();
|
|
|
|
|
} else {
|
|
|
|
|
this._trayIcon.click(event);
|
|
|
|
|
}
|
2011-01-13 15:04:37 -05:00
|
|
|
|
return true;
|
|
|
|
|
},
|
|
|
|
|
|
2010-08-09 12:33:34 -04:00
|
|
|
|
_setApp: function() {
|
2010-12-16 15:49:47 -05:00
|
|
|
|
if (this.app)
|
|
|
|
|
return;
|
|
|
|
|
|
2010-08-09 12:33:34 -04:00
|
|
|
|
this.app = Shell.WindowTracker.get_default().get_app_from_pid(this._pid);
|
|
|
|
|
if (!this.app)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
// Only override the icon if we were previously using
|
2011-01-07 10:22:28 -05:00
|
|
|
|
// notification-based icons (ie, not a trayicon) or if it was unset before
|
2011-01-13 15:04:37 -05:00
|
|
|
|
if (!this._trayIcon) {
|
2010-08-09 12:33:34 -04:00
|
|
|
|
this.useNotificationIcon = false;
|
|
|
|
|
this._setSummaryIcon(this.app.create_icon_texture (this.ICON_SIZE));
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
setTrayIcon: function(icon) {
|
|
|
|
|
this._setSummaryIcon(icon);
|
|
|
|
|
this.useNotificationIcon = false;
|
2011-01-13 15:04:37 -05:00
|
|
|
|
this._trayIcon = icon;
|
2010-08-09 12:33:34 -04:00
|
|
|
|
},
|
|
|
|
|
|
2011-02-11 14:43:01 -05:00
|
|
|
|
open: function(notification) {
|
2011-03-21 17:43:34 -04:00
|
|
|
|
this.destroyNonResidentNotifications();
|
2010-02-15 12:27:28 -05:00
|
|
|
|
this.openApp();
|
2010-09-30 16:56:50 -04:00
|
|
|
|
},
|
|
|
|
|
|
2011-03-21 17:43:34 -04:00
|
|
|
|
_lastNotificationRemoved: function() {
|
2011-01-13 15:04:37 -05:00
|
|
|
|
if (!this._trayIcon)
|
2010-09-30 16:56:50 -04:00
|
|
|
|
this.destroy();
|
2010-02-15 12:27:28 -05:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
openApp: function() {
|
2010-09-02 10:45:53 -04:00
|
|
|
|
if (this.app == null)
|
2010-02-15 12:27:28 -05:00
|
|
|
|
return;
|
2010-09-02 10:45:53 -04:00
|
|
|
|
|
2010-02-15 12:27:28 -05:00
|
|
|
|
let windows = this.app.get_windows();
|
|
|
|
|
if (windows.length > 0) {
|
|
|
|
|
let mostRecentWindow = windows[0];
|
|
|
|
|
Main.activateWindow(mostRecentWindow);
|
|
|
|
|
}
|
2011-03-07 18:10:10 -05:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
destroy: function() {
|
|
|
|
|
MessageTray.Source.prototype.destroy.call(this);
|
2010-01-13 15:05:20 -05:00
|
|
|
|
}
|
|
|
|
|
};
|