如何编写没有window class的QT系统托盘应用程序,并将其与另一个进程集成?
How to write QT system tray app without a window class, and integrate it with another process?
这是我的设置:
保持 运行ning 并完成其工作的后台进程。
启动上述进程并监视它的启动器,如果它崩溃或被杀死则重新启动它。
我希望添加对启动器进程的系统托盘访问(理想情况下,启动器进程将包含系统托盘显示代码)并启用从系统托盘上下文菜单触发的基本选项(启动、停止等) .系统托盘不需要自己的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的任务。
鉴于评论中的说明,关于如何获取上下文菜单或激活操作调用的代码,您有几个选项。
一个接收器对象:基本上是示例使用的内容,只是您不从任何 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);
连接到 stand-alone 个函数
void onActivated(QSystemTrayIcon::ActivationReason reason);
connect(trayIcon, &QSystemTrayIcon::activated, &onActivated);
在支持 C++11 的环境中,连接到 lambda
connect(trayIcon, &QSystemTrayIcon::activated,
[](QSystemTrayIcon::ActivationReason reason) {});
对于上下文菜单,同样的技术适用,"sender" 对象是您添加到菜单的 QAction
项,它们的信号是 triggered()
或 toggled(bool)
,具体取决于关于操作是否可以直接单击或在 "on" 和 "off" 状态之间切换。
这是我的设置:
保持 运行ning 并完成其工作的后台进程。
启动上述进程并监视它的启动器,如果它崩溃或被杀死则重新启动它。
我希望添加对启动器进程的系统托盘访问(理想情况下,启动器进程将包含系统托盘显示代码)并启用从系统托盘上下文菜单触发的基本选项(启动、停止等) .系统托盘不需要自己的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的任务。
鉴于评论中的说明,关于如何获取上下文菜单或激活操作调用的代码,您有几个选项。
一个接收器对象:基本上是示例使用的内容,只是您不从任何 window 类型派生接收器 class。 对于基于宏的 signal/slot 连接,基类型需要是
QObject
或从中派生的东西,对于基于函数指针的连接,它可以是任何 classclass 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);
连接到 stand-alone 个函数
void onActivated(QSystemTrayIcon::ActivationReason reason); connect(trayIcon, &QSystemTrayIcon::activated, &onActivated);
在支持 C++11 的环境中,连接到 lambda
connect(trayIcon, &QSystemTrayIcon::activated, [](QSystemTrayIcon::ActivationReason reason) {});
对于上下文菜单,同样的技术适用,"sender" 对象是您添加到菜单的 QAction
项,它们的信号是 triggered()
或 toggled(bool)
,具体取决于关于操作是否可以直接单击或在 "on" 和 "off" 状态之间切换。