如何编写没有window class的QT系统托盘应用程序,并将其与另一个进程集成?

How to write QT system tray app without a window class, and integrate it with another process?

这是我的设置:

我希望添加对启动器进程的系统托盘访问(理想情况下,启动器进程将包含系统托盘显示代码)并启用从系统托盘上下文菜单触发的基本选项(启动、停止等) .系统托盘不需要自己的window。只是一个 windowless 系统托盘,带有包含 2-3 个选项的上下文菜单。

由于到目前为止编写的所有代码都在 C/C++ 中,我需要它在 Windows 和 Linux 上 运行,QT 是显而易见的选择.我发现通过基本的 QT 启动器托盘显示非常令人沮丧。我见过的几乎每个 QSystemTrayIcon 示例都包含 'mainwindow' 继承。

下面是我用来创建系统托盘的代码。

#include <QtWidgets/QApplication>
#include <QtCore/QDebug>
#include <QtGui/QIcon>
#include <QtWidgets/QSystemTrayIcon>
#include <QtWidgets/QMainWindow>
#include <QtWidgets/QMenu>


int main(int argc, char **argv)
{
    QApplication app(argc, argv);
    QPixmap oPixmap(32,32);
    //QMenu* menu1 = new QMenu(); // want to get a context menu from system tray
    oPixmap.load ("systemTrayIcon.png");

    QIcon oIcon( oPixmap );

    QSystemTrayIcon *trayIcon = new QSystemTrayIcon(oIcon);
    qDebug() << trayIcon->isSystemTrayAvailable();
    trayIcon->setContextMenu( menu1);
    trayIcon->setVisible(true);
    trayIcon->showMessage("Test Message", "Text", QSystemTrayIcon::Information, 1000);

    return app.exec();
}

代码可以很好地显示系统托盘,但我一直无法解决如何向其添加菜单的问题。 我想要的是:

1) 将上下文菜单添加到上面的系统托盘中,而不添加任何 window class(除非不可能)

2) 将这些上下文菜单项连接到我现有代码中的函数

3) app.exec() 似乎是一个处理QT 事件的无限循环。然而,由于我的启动器有它自己的事件循环,我想使 QT 事件循环与我的启动器循环集成在一起。也就是说,在事件循环中加入一些非QT的任务。

鉴于评论中的说明,关于如何获取上下文菜单或激活操作调用的代码,您有几个选项。

  1. 一个接收器对象:基本上是示例使用的内容,只是您不从任何 window 类型派生接收器 class。 对于基于宏的 signal/slot 连接,基类型需要是 QObject 或从中派生的东西,对于基于函数指针的连接,它可以是任何 class

    class MyReceiver : public QObject
    {
        Q_OBJECT
    public slots:
        void onActivated(QSystemTrayIcon::ActivationReason reason);
    };
    
    
    // in main()
    MyReceiver receiver;
    
    // macro based connect
    connect(trayIcon, SIGNAL(activated(QSystemTrayIcon::ActivationReason)),
            &receiver, SLOT(onActivated(QSystemTrayIcon::ActivationReason)));
    
    
    // or function pointer based connect
    connect(trayIcon, &QSystemTrayIcon::activated,
            &receiver, &MyReceiver::onActivated);
    
  2. 连接到 stand-alone 个函数

    void onActivated(QSystemTrayIcon::ActivationReason reason);
    
    
    connect(trayIcon, &QSystemTrayIcon::activated, &onActivated);
    
  3. 在支持 C++11 的环境中,连接到 lambda

    connect(trayIcon, &QSystemTrayIcon::activated,
            [](QSystemTrayIcon::ActivationReason reason) {});
    

对于上下文菜单,同样的技术适用,"sender" 对象是您添加到菜单的 QAction 项,它们的信号是 triggered()toggled(bool),具体取决于关于操作是否可以直接单击或在 "on" 和 "off" 状态之间切换。