/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the QtWidgets module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 3 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL3 included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 3 requirements ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 2.0 or (at your option) the GNU General ** Public license version 3 or any later version approved by the KDE Free ** Qt Foundation. The licenses are as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3 ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-2.0.html and ** https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include "qtwidgetsglobal.h" #if QT_CONFIG(label) #include "qlabel.h" #endif #include "qpainter.h" #include "qpixmap.h" #include "qbitmap.h" #include "qevent.h" #include "qapplication.h" #include "qlist.h" #if QT_CONFIG(menu) #include "qmenu.h" #endif #include "qtimer.h" #include "qsystemtrayicon_p.h" #include "qpaintengine.h" #include #include #include #include #include #include #include #include #include #include #include #ifndef QT_NO_SYSTEMTRAYICON QT_BEGIN_NAMESPACE static inline unsigned long locateSystemTray() { return (unsigned long)QGuiApplication::platformNativeInterface()->nativeResourceForScreen(QByteArrayLiteral("traywindow"), QGuiApplication::primaryScreen()); } // System tray widget. Could be replaced by a QWindow with // a backing store if it did not need tooltip handling. class QSystemTrayIconSys : public QWidget { Q_OBJECT public: explicit QSystemTrayIconSys(QSystemTrayIcon *q); inline void updateIcon() { update(); } inline QSystemTrayIcon *systemTrayIcon() const { return q; } QRect globalGeometry() const; protected: virtual void mousePressEvent(QMouseEvent *ev) override; virtual void mouseDoubleClickEvent(QMouseEvent *ev) override; virtual bool event(QEvent *) override; virtual void paintEvent(QPaintEvent *) override; virtual void resizeEvent(QResizeEvent *) override; virtual void moveEvent(QMoveEvent *) override; private slots: void systemTrayWindowChanged(QScreen *screen); private: bool addToTray(); QSystemTrayIcon *q; QPixmap background; }; QSystemTrayIconSys::QSystemTrayIconSys(QSystemTrayIcon *qIn) : QWidget(0, Qt::Window | Qt::FramelessWindowHint | Qt::BypassWindowManagerHint) , q(qIn) { setObjectName(QStringLiteral("QSystemTrayIconSys")); #if QT_CONFIG(tooltip) setToolTip(q->toolTip()); #endif setAttribute(Qt::WA_AlwaysShowToolTips, true); setAttribute(Qt::WA_QuitOnClose, false); const QSize size(22, 22); // Gnome, standard size setGeometry(QRect(QPoint(0, 0), size)); setMinimumSize(size); // We need two different behaviors depending on whether the X11 visual for the system tray // (a) exists and (b) supports an alpha channel, i.e. is 32 bits. // If we have a visual that has an alpha channel, we can paint this widget with a transparent // background and it will work. // However, if there's no alpha channel visual, in order for transparent tray icons to work, // we do not have a transparent background on the widget, but set the BackPixmap property of our // window to ParentRelative (so that it inherits the background of its X11 parent window), call // xcb_clear_region before painting (so that the inherited background is visible) and then grab // the just-drawn background from the X11 server. bool hasAlphaChannel = QXcbIntegrationFunctions::xEmbedSystemTrayVisualHasAlphaChannel(); setAttribute(Qt::WA_TranslucentBackground, hasAlphaChannel); if (!hasAlphaChannel) { createWinId(); QXcbWindowFunctions::setParentRelativeBackPixmap(windowHandle()); // XXX: This is actually required, but breaks things ("QWidget::paintEngine: Should no // longer be called"). Why is this needed? When the widget is drawn, we use tricks to grab // the tray icon's background from the server. If the tray icon isn't visible (because // another window is on top of it), the trick fails and instead uses the content of that // other window as the background. // setAttribute(Qt::WA_PaintOnScreen); } addToTray(); } bool QSystemTrayIconSys::addToTray() { if (!locateSystemTray()) return false; createWinId(); setMouseTracking(true); if (!QXcbWindowFunctions::requestSystemTrayWindowDock(windowHandle())) return false; if (!background.isNull()) background = QPixmap(); show(); return true; } void QSystemTrayIconSys::systemTrayWindowChanged(QScreen *) { if (locateSystemTray()) { addToTray(); } else { QBalloonTip::hideBalloon(); hide(); // still no luck destroy(); } } QRect QSystemTrayIconSys::globalGeometry() const { return QXcbWindowFunctions::systemTrayWindowGlobalGeometry(windowHandle()); } void QSystemTrayIconSys::mousePressEvent(QMouseEvent *ev) { QPoint globalPos = ev->globalPos(); #ifndef QT_NO_CONTEXTMENU if (ev->button() == Qt::RightButton && q->contextMenu()) q->contextMenu()->popup(globalPos); #else Q_UNUSED(globalPos) #endif // QT_NO_CONTEXTMENU if (QBalloonTip::isBalloonVisible()) { emit q->messageClicked(); QBalloonTip::hideBalloon(); } if (ev->button() == Qt::LeftButton) emit q->activated(QSystemTrayIcon::Trigger); else if (ev->button() == Qt::RightButton) emit q->activated(QSystemTrayIcon::Context); else if (ev->button() == Qt::MidButton) emit q->activated(QSystemTrayIcon::MiddleClick); } void QSystemTrayIconSys::mouseDoubleClickEvent(QMouseEvent *ev) { if (ev->button() == Qt::LeftButton) emit q->activated(QSystemTrayIcon::DoubleClick); } bool QSystemTrayIconSys::event(QEvent *e) { switch (e->type()) { case QEvent::ToolTip: QApplication::sendEvent(q, e); break; #if QT_CONFIG(wheelevent) case QEvent::Wheel: return QApplication::sendEvent(q, e); #endif default: break; } return QWidget::event(e); } void QSystemTrayIconSys::paintEvent(QPaintEvent *) { const QRect rect(QPoint(0, 0), geometry().size()); QPainter painter(this); // If we have Qt::WA_TranslucentBackground set, during widget creation // we detected the systray visual supported an alpha channel if (testAttribute(Qt::WA_TranslucentBackground)) { painter.setCompositionMode(QPainter::CompositionMode_Source); painter.fillRect(rect, Qt::transparent); } else { // clearRegion() was called on XEMBED_EMBEDDED_NOTIFY, so we hope that got done by now. // Grab the tray background pixmap, before rendering the icon for the first time. if (background.isNull()) { background = QGuiApplication::primaryScreen()->grabWindow(winId(), 0, 0, rect.size().width(), rect.size().height()); } // Then paint over the icon area with the background before compositing the icon on top. painter.drawPixmap(QPoint(0, 0), background); } painter.setCompositionMode(QPainter::CompositionMode_SourceOver); q->icon().paint(&painter, rect); } void QSystemTrayIconSys::moveEvent(QMoveEvent *event) { QWidget::moveEvent(event); if (QBalloonTip::isBalloonVisible()) QBalloonTip::updateBalloonPosition(globalGeometry().center()); } void QSystemTrayIconSys::resizeEvent(QResizeEvent *event) { update(); QWidget::resizeEvent(event); if (QBalloonTip::isBalloonVisible()) QBalloonTip::updateBalloonPosition(globalGeometry().center()); } //////////////////////////////////////////////////////////////////////////// QSystemTrayIconPrivate::QSystemTrayIconPrivate() : sys(0), qpa_sys(QGuiApplicationPrivate::platformTheme()->createPlatformSystemTrayIcon()), visible(false) { } QSystemTrayIconPrivate::~QSystemTrayIconPrivate() { delete qpa_sys; } void QSystemTrayIconPrivate::install_sys() { if (qpa_sys) { install_sys_qpa(); return; } Q_Q(QSystemTrayIcon); if (!sys && locateSystemTray()) { sys = new QSystemTrayIconSys(q); QObject::connect(QGuiApplication::platformNativeInterface(), SIGNAL(systemTrayWindowChanged(QScreen*)), sys, SLOT(systemTrayWindowChanged(QScreen*))); } } QRect QSystemTrayIconPrivate::geometry_sys() const { if (qpa_sys) return qpa_sys->geometry(); if (!sys) return QRect(); return sys->globalGeometry(); } void QSystemTrayIconPrivate::remove_sys() { if (qpa_sys) { remove_sys_qpa(); return; } if (!sys) return; QBalloonTip::hideBalloon(); sys->hide(); // this should do the trick, but... delete sys; // wm may resize system tray only for DestroyEvents sys = 0; } void QSystemTrayIconPrivate::updateIcon_sys() { if (qpa_sys) { qpa_sys->updateIcon(icon); return; } if (sys) sys->updateIcon(); } void QSystemTrayIconPrivate::updateMenu_sys() { #if QT_CONFIG(menu) if (qpa_sys && menu) { addPlatformMenu(menu); qpa_sys->updateMenu(menu->platformMenu()); } #endif } void QSystemTrayIconPrivate::updateToolTip_sys() { if (qpa_sys) { qpa_sys->updateToolTip(toolTip); return; } if (!sys) return; #ifndef QT_NO_TOOLTIP sys->setToolTip(toolTip); #endif } bool QSystemTrayIconPrivate::isSystemTrayAvailable_sys() { QScopedPointer sys(QGuiApplicationPrivate::platformTheme()->createPlatformSystemTrayIcon()); if (sys && sys->isSystemTrayAvailable()) return true; // no QPlatformSystemTrayIcon so fall back to default xcb platform behavior const QString platform = QGuiApplication::platformName(); if (platform.compare(QLatin1String("xcb"), Qt::CaseInsensitive) == 0) return locateSystemTray(); return false; } bool QSystemTrayIconPrivate::supportsMessages_sys() { QScopedPointer sys(QGuiApplicationPrivate::platformTheme()->createPlatformSystemTrayIcon()); if (sys) return sys->supportsMessages(); // no QPlatformSystemTrayIcon so fall back to default xcb platform behavior return true; } void QSystemTrayIconPrivate::showMessage_sys(const QString &title, const QString &message, const QIcon &icon, QSystemTrayIcon::MessageIcon msgIcon, int msecs) { if (qpa_sys) { qpa_sys->showMessage(title, message, icon, static_cast(msgIcon), msecs); return; } if (!sys) return; QBalloonTip::showBalloon(icon, title, message, sys->systemTrayIcon(), sys->globalGeometry().center(), msecs); } QT_END_NAMESPACE #include "qsystemtrayicon_x11.moc" #endif //QT_NO_SYSTEMTRAYICON