PyQt5:如何从系统托盘图标上下文菜单启动窗口?

Posted

技术标签:

【中文标题】PyQt5:如何从系统托盘图标上下文菜单启动窗口?【英文标题】:PyQt5: how do I launch a window from a system tray icon context menu? 【发布时间】:2018-11-24 16:26:07 【问题描述】:

我有两个单独的文件,一个用于创建系统托盘图标和上下文菜单,另一个用于接收用户输入。

traywindow.py 包含以下内容:

import sys
import os
from PyQt5 import QtCore, QtWidgets
from PyQt5.QtWidgets import QMainWindow, QWidget, QLabel, QLineEdit
from PyQt5.QtWidgets import QPushButton
from PyQt5.QtCore import QSize    

class MainWindow(QMainWindow):
    def __init__(self):
        QMainWindow.__init__(self)

        self.setMinimumSize(QSize(320, 172))    
        self.setWindowTitle("Set Account") 

        # Setup username field
        self.username_label = QLabel(self)
        self.username_label.setText('Username')
        self.username_field = QLineEdit(self)
        self.username_label.move(45, 20)
        self.username_field.move(115, 23)
        self.username_field.resize(150, 25)

        # Setup OK button
        pybutton = QPushButton('OK', self)
        pybutton.clicked.connect(self.buttonClicked)
        pybutton.resize(100,32)
        pybutton.move(110, 128)        

    def buttonClicked(self):
        print('Username: ' + self.username_field.text())

def displayWindow():
    app = QtWidgets.QApplication(sys.argv)
    preferences_window = MainWindow()
    preferences_window.show()
    sys.exit(app.exec_())

if __name__ == "__main__":
    displayWindow()

产生:

trayapp.py 包含以下内容:

import os
import sys
import traywindow
from PyQt5 import QtWidgets, QtCore, QtGui

class SystemTrayIcon(QtWidgets.QSystemTrayIcon):

    def __init__(self, icon, parent=None):
        QtWidgets.QSystemTrayIcon.__init__(self, icon, parent)

        menu = QtWidgets.QMenu(parent)
        prefs_action = menu.addAction('Preferences...')
        self.setContextMenu(menu)
        prefs_action.triggered.connect(self.setPrefs)

    def setPrefs(self):
        traywindow.displayWindow()

def main(image):
    app = QtWidgets.QApplication(sys.argv)
    w = QtWidgets.QWidget()
    trayIcon = SystemTrayIcon(QtGui.QIcon(image), w)
    trayIcon.show()
    sys.exit(app.exec_())

if __name__ == '__main__':
    icon = 'icon.png'
    main(icon)

产生:

我想从上下文菜单中启动窗口,所以当我点击Preferences... 时,它将打开Set Account 窗口,当我填写字段并点击OK 时,将其捕获到变量/将这些作为参数传递到trayapp.py 文件中。目前,当我点击Preferences... 时,我在上面的代码中的尝试给了我这个:

Traceback (most recent call last):
  File "/Users/Username/Downloads/trayapp.py", line 17, in setPrefs
    traywindow.displayWindow()
  File "/Users/Username/Downloads/traywindow.py", line 36, in displayWindow
    sys.exit(app.exec_())
SystemExit: -1
[Finished in 3.0s with exit code -6]

我是 PyQt5 的新手,感觉我缺少一些基本的东西。从错误来看,我认为这与最后调用每个文件以生成其 UI 的方式有关,但到目前为止我还没有在文档中找到答案。

【问题讨论】:

【参考方案1】:

你只能有一个 QApplication 并且在 displayWindow 中你正在创建第二个 QApplication,而不是创建一个 MainWindow 并且不要调用 displayWindow。

def setPrefs(self):
    self.window = traywindow.MainWindow()
    self.window.show()

【讨论】:

以上是关于PyQt5:如何从系统托盘图标上下文菜单启动窗口?的主要内容,如果未能解决你的问题,请参考以下文章

WinForm 之 窗口最小化到托盘及右键图标显示菜单

PyQt5,MacOS - 强制 GUI 坚持当前打开的桌面空间

实现系统托盘

centos7.9下Qt 实现系统托盘,托盘菜单,托盘消息

OSX 上的 QT:托盘图标 - 图标 Dock 问题

PyQt5托盘图标消失