2016-12-31 11:01:21 -07:00
|
|
|
// Copyright (c) 2011-2016 The Bitcoin Core developers
|
2014-12-13 12:09:33 +08:00
|
|
|
// Distributed under the MIT software license, see the accompanying
|
2013-11-04 16:20:43 +01:00
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
2011-09-03 20:52:54 +02:00
|
|
|
#include "notificator.h"
|
|
|
|
|
2013-07-23 08:52:24 +02:00
|
|
|
#include <QApplication>
|
2013-04-13 00:13:08 -05:00
|
|
|
#include <QByteArray>
|
2011-09-03 20:52:54 +02:00
|
|
|
#include <QIcon>
|
2013-04-13 00:13:08 -05:00
|
|
|
#include <QImageWriter>
|
|
|
|
#include <QMessageBox>
|
|
|
|
#include <QMetaType>
|
2011-09-03 20:52:54 +02:00
|
|
|
#include <QStyle>
|
|
|
|
#include <QSystemTrayIcon>
|
2011-10-07 13:21:45 +02:00
|
|
|
#include <QTemporaryFile>
|
2013-04-13 00:13:08 -05:00
|
|
|
#include <QVariant>
|
2011-09-24 11:56:33 +02:00
|
|
|
#ifdef USE_DBUS
|
2011-09-03 20:52:54 +02:00
|
|
|
#include <stdint.h>
|
2013-04-13 00:13:08 -05:00
|
|
|
#include <QtDBus>
|
2011-10-07 13:21:45 +02:00
|
|
|
#endif
|
2013-12-18 20:29:24 -05:00
|
|
|
// Include ApplicationServices.h after QtDbus to avoid redefinition of check().
|
|
|
|
// This affects at least OSX 10.6. See /usr/include/AssertMacros.h for details.
|
|
|
|
// Note: This could also be worked around using:
|
|
|
|
// #define __ASSERT_MACROS_DEFINE_VERSIONS_WITHOUT_UNDERSCORES 0
|
|
|
|
#ifdef Q_OS_MAC
|
|
|
|
#include <ApplicationServices/ApplicationServices.h>
|
|
|
|
#include "macnotificationhandler.h"
|
|
|
|
#endif
|
2011-10-07 13:21:45 +02:00
|
|
|
|
2013-04-13 00:13:08 -05:00
|
|
|
|
2014-05-23 13:58:12 -03:00
|
|
|
#ifdef USE_DBUS
|
2011-09-03 20:52:54 +02:00
|
|
|
// https://wiki.ubuntu.com/NotificationDevelopmentGuidelines recommends at least 128
|
|
|
|
const int FREEDESKTOP_NOTIFICATION_ICON_SIZE = 128;
|
2014-05-23 13:58:12 -03:00
|
|
|
#endif
|
2011-09-03 20:52:54 +02:00
|
|
|
|
2016-09-09 13:43:29 +02:00
|
|
|
Notificator::Notificator(const QString &_programName, QSystemTrayIcon *_trayIcon, QWidget *_parent) :
|
|
|
|
QObject(_parent),
|
|
|
|
parent(_parent),
|
|
|
|
programName(_programName),
|
2011-09-03 20:52:54 +02:00
|
|
|
mode(None),
|
2016-09-09 13:43:29 +02:00
|
|
|
trayIcon(_trayIcon)
|
2011-09-24 11:56:33 +02:00
|
|
|
#ifdef USE_DBUS
|
2011-09-03 20:52:54 +02:00
|
|
|
,interface(0)
|
|
|
|
#endif
|
|
|
|
{
|
2016-09-09 13:43:29 +02:00
|
|
|
if(_trayIcon && _trayIcon->supportsMessages())
|
2011-09-03 20:52:54 +02:00
|
|
|
{
|
|
|
|
mode = QSystemTray;
|
|
|
|
}
|
2011-09-24 11:56:33 +02:00
|
|
|
#ifdef USE_DBUS
|
2011-09-03 20:52:54 +02:00
|
|
|
interface = new QDBusInterface("org.freedesktop.Notifications",
|
2013-12-03 09:25:24 +01:00
|
|
|
"/org/freedesktop/Notifications", "org.freedesktop.Notifications");
|
2011-09-03 20:52:54 +02:00
|
|
|
if(interface->isValid())
|
|
|
|
{
|
|
|
|
mode = Freedesktop;
|
|
|
|
}
|
|
|
|
#endif
|
2012-09-21 19:06:53 +02:00
|
|
|
#ifdef Q_OS_MAC
|
2013-05-10 22:20:51 +02:00
|
|
|
// check if users OS has support for NSUserNotification
|
|
|
|
if( MacNotificationHandler::instance()->hasUserNotificationCenterSupport()) {
|
|
|
|
mode = UserNotificationCenter;
|
|
|
|
}
|
2011-10-07 13:21:45 +02:00
|
|
|
#endif
|
2011-09-03 20:52:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Notificator::~Notificator()
|
|
|
|
{
|
2011-09-24 11:56:33 +02:00
|
|
|
#ifdef USE_DBUS
|
2011-09-03 20:52:54 +02:00
|
|
|
delete interface;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2011-09-24 11:56:33 +02:00
|
|
|
#ifdef USE_DBUS
|
2011-09-03 20:52:54 +02:00
|
|
|
|
|
|
|
// Loosely based on http://www.qtcentre.org/archive/index.php/t-25879.html
|
|
|
|
class FreedesktopImage
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
FreedesktopImage() {}
|
2017-08-01 12:22:41 +02:00
|
|
|
explicit FreedesktopImage(const QImage &img);
|
2011-09-03 20:52:54 +02:00
|
|
|
|
|
|
|
static int metaType();
|
|
|
|
|
2012-07-26 00:48:39 +00:00
|
|
|
// Image to variant that can be marshalled over DBus
|
2011-09-03 20:52:54 +02:00
|
|
|
static QVariant toVariant(const QImage &img);
|
|
|
|
|
|
|
|
private:
|
|
|
|
int width, height, stride;
|
|
|
|
bool hasAlpha;
|
|
|
|
int channels;
|
|
|
|
int bitsPerSample;
|
|
|
|
QByteArray image;
|
|
|
|
|
|
|
|
friend QDBusArgument &operator<<(QDBusArgument &a, const FreedesktopImage &i);
|
|
|
|
friend const QDBusArgument &operator>>(const QDBusArgument &a, FreedesktopImage &i);
|
|
|
|
};
|
|
|
|
|
|
|
|
Q_DECLARE_METATYPE(FreedesktopImage);
|
|
|
|
|
|
|
|
// Image configuration settings
|
|
|
|
const int CHANNELS = 4;
|
|
|
|
const int BYTES_PER_PIXEL = 4;
|
|
|
|
const int BITS_PER_SAMPLE = 8;
|
|
|
|
|
|
|
|
FreedesktopImage::FreedesktopImage(const QImage &img):
|
|
|
|
width(img.width()),
|
|
|
|
height(img.height()),
|
|
|
|
stride(img.width() * BYTES_PER_PIXEL),
|
|
|
|
hasAlpha(true),
|
|
|
|
channels(CHANNELS),
|
|
|
|
bitsPerSample(BITS_PER_SAMPLE)
|
|
|
|
{
|
|
|
|
// Convert 00xAARRGGBB to RGBA bytewise (endian-independent) format
|
|
|
|
QImage tmp = img.convertToFormat(QImage::Format_ARGB32);
|
2013-02-09 19:18:53 +01:00
|
|
|
const uint32_t *data = reinterpret_cast<const uint32_t*>(tmp.bits());
|
2011-09-03 20:52:54 +02:00
|
|
|
|
|
|
|
unsigned int num_pixels = width * height;
|
|
|
|
image.resize(num_pixels * BYTES_PER_PIXEL);
|
|
|
|
|
|
|
|
for(unsigned int ptr = 0; ptr < num_pixels; ++ptr)
|
|
|
|
{
|
|
|
|
image[ptr*BYTES_PER_PIXEL+0] = data[ptr] >> 16; // R
|
|
|
|
image[ptr*BYTES_PER_PIXEL+1] = data[ptr] >> 8; // G
|
|
|
|
image[ptr*BYTES_PER_PIXEL+2] = data[ptr]; // B
|
|
|
|
image[ptr*BYTES_PER_PIXEL+3] = data[ptr] >> 24; // A
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QDBusArgument &operator<<(QDBusArgument &a, const FreedesktopImage &i)
|
|
|
|
{
|
|
|
|
a.beginStructure();
|
|
|
|
a << i.width << i.height << i.stride << i.hasAlpha << i.bitsPerSample << i.channels << i.image;
|
|
|
|
a.endStructure();
|
|
|
|
return a;
|
|
|
|
}
|
|
|
|
|
|
|
|
const QDBusArgument &operator>>(const QDBusArgument &a, FreedesktopImage &i)
|
|
|
|
{
|
|
|
|
a.beginStructure();
|
|
|
|
a >> i.width >> i.height >> i.stride >> i.hasAlpha >> i.bitsPerSample >> i.channels >> i.image;
|
|
|
|
a.endStructure();
|
|
|
|
return a;
|
|
|
|
}
|
|
|
|
|
|
|
|
int FreedesktopImage::metaType()
|
|
|
|
{
|
|
|
|
return qDBusRegisterMetaType<FreedesktopImage>();
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant FreedesktopImage::toVariant(const QImage &img)
|
|
|
|
{
|
|
|
|
FreedesktopImage fimg(img);
|
|
|
|
return QVariant(FreedesktopImage::metaType(), &fimg);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Notificator::notifyDBus(Class cls, const QString &title, const QString &text, const QIcon &icon, int millisTimeout)
|
|
|
|
{
|
|
|
|
Q_UNUSED(cls);
|
|
|
|
// Arguments for DBus call:
|
|
|
|
QList<QVariant> args;
|
|
|
|
|
|
|
|
// Program Name:
|
|
|
|
args.append(programName);
|
|
|
|
|
|
|
|
// Unique ID of this notification type:
|
|
|
|
args.append(0U);
|
|
|
|
|
|
|
|
// Application Icon, empty string
|
|
|
|
args.append(QString());
|
|
|
|
|
|
|
|
// Summary
|
|
|
|
args.append(title);
|
|
|
|
|
|
|
|
// Body
|
|
|
|
args.append(text);
|
|
|
|
|
|
|
|
// Actions (none, actions are deprecated)
|
|
|
|
QStringList actions;
|
|
|
|
args.append(actions);
|
|
|
|
|
|
|
|
// Hints
|
|
|
|
QVariantMap hints;
|
|
|
|
|
|
|
|
// If no icon specified, set icon based on class
|
|
|
|
QIcon tmpicon;
|
|
|
|
if(icon.isNull())
|
|
|
|
{
|
|
|
|
QStyle::StandardPixmap sicon = QStyle::SP_MessageBoxQuestion;
|
|
|
|
switch(cls)
|
|
|
|
{
|
|
|
|
case Information: sicon = QStyle::SP_MessageBoxInformation; break;
|
|
|
|
case Warning: sicon = QStyle::SP_MessageBoxWarning; break;
|
|
|
|
case Critical: sicon = QStyle::SP_MessageBoxCritical; break;
|
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
tmpicon = QApplication::style()->standardIcon(sicon);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
tmpicon = icon;
|
|
|
|
}
|
|
|
|
hints["icon_data"] = FreedesktopImage::toVariant(tmpicon.pixmap(FREEDESKTOP_NOTIFICATION_ICON_SIZE).toImage());
|
|
|
|
args.append(hints);
|
|
|
|
|
|
|
|
// Timeout (in msec)
|
|
|
|
args.append(millisTimeout);
|
|
|
|
|
|
|
|
// "Fire and forget"
|
|
|
|
interface->callWithArgumentList(QDBus::NoBlock, "Notify", args);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void Notificator::notifySystray(Class cls, const QString &title, const QString &text, const QIcon &icon, int millisTimeout)
|
|
|
|
{
|
|
|
|
Q_UNUSED(icon);
|
|
|
|
QSystemTrayIcon::MessageIcon sicon = QSystemTrayIcon::NoIcon;
|
|
|
|
switch(cls) // Set icon based on class
|
|
|
|
{
|
|
|
|
case Information: sicon = QSystemTrayIcon::Information; break;
|
|
|
|
case Warning: sicon = QSystemTrayIcon::Warning; break;
|
|
|
|
case Critical: sicon = QSystemTrayIcon::Critical; break;
|
|
|
|
}
|
|
|
|
trayIcon->showMessage(title, text, sicon, millisTimeout);
|
|
|
|
}
|
|
|
|
|
2011-10-07 13:21:45 +02:00
|
|
|
// Based on Qt's tray icon implementation
|
2012-09-21 19:06:53 +02:00
|
|
|
#ifdef Q_OS_MAC
|
2013-05-10 22:20:51 +02:00
|
|
|
void Notificator::notifyMacUserNotificationCenter(Class cls, const QString &title, const QString &text, const QIcon &icon) {
|
|
|
|
// icon is not supported by the user notification center yet. OSX will use the app icon.
|
|
|
|
MacNotificationHandler::instance()->showNotification(title, text);
|
2011-10-07 13:21:45 +02:00
|
|
|
}
|
2013-05-10 22:20:51 +02:00
|
|
|
|
2011-10-07 13:21:45 +02:00
|
|
|
#endif
|
|
|
|
|
2011-09-03 20:52:54 +02:00
|
|
|
void Notificator::notify(Class cls, const QString &title, const QString &text, const QIcon &icon, int millisTimeout)
|
|
|
|
{
|
|
|
|
switch(mode)
|
|
|
|
{
|
2011-09-24 11:56:33 +02:00
|
|
|
#ifdef USE_DBUS
|
2011-09-03 20:52:54 +02:00
|
|
|
case Freedesktop:
|
|
|
|
notifyDBus(cls, title, text, icon, millisTimeout);
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
case QSystemTray:
|
|
|
|
notifySystray(cls, title, text, icon, millisTimeout);
|
|
|
|
break;
|
2012-09-21 19:06:53 +02:00
|
|
|
#ifdef Q_OS_MAC
|
2013-05-10 22:20:51 +02:00
|
|
|
case UserNotificationCenter:
|
|
|
|
notifyMacUserNotificationCenter(cls, title, text, icon);
|
|
|
|
break;
|
2011-10-07 13:21:45 +02:00
|
|
|
#endif
|
2011-09-03 20:52:54 +02:00
|
|
|
default:
|
|
|
|
if(cls == Critical)
|
|
|
|
{
|
2012-07-26 00:48:39 +00:00
|
|
|
// Fall back to old fashioned pop-up dialog if critical and no other notification available
|
2011-09-03 20:52:54 +02:00
|
|
|
QMessageBox::critical(parent, title, text, QMessageBox::Ok, QMessageBox::Ok);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|