PyQT5 qsytemtrayicon

PyQT5 qsytemtrayicon

我的 qsystemtray 有问题。当我 运行 在没有 "while" 的情况下编写此代码时,我可以看到图标。 with while I can't watch icon(我检查程序进入条件"if (a.inbox.unread_count)> 0"),但是icon必须不是available.That的区域是,托盘图标出现,但没有绘制。有什么方法可以解决这个问题吗? OS: Astra-linux(fork debian)

# -*- coding: utf-8 -*-

# Form implementation generated from reading ui file 'qttray.ui'
#
# Created by: PyQt5 UI code generator 5.10.1
#
# WARNING! All changes made in this file will be lost!

from PyQt5 import QtCore, QtGui, QtWidgets
import extest4
import time
class Ui_MainWindow(object):
    def setupUi(self, MainWindow):
        MainWindow.setObjectName("MainWindow")
        MainWindow.resize(1014, 680)
        self.centralwidget = QtWidgets.QWidget(MainWindow)
        self.centralwidget.setMaximumSize(QtCore.QSize(800, 547))
        self.centralwidget.setObjectName("centralwidget")
        self.horizontalLayout_2 = QtWidgets.QHBoxLayout(self.centralwidget)
        self.horizontalLayout_2.setObjectName("horizontalLayout_2")
        self.horizontalLayout = QtWidgets.QHBoxLayout()
        self.horizontalLayout.setObjectName("horizontalLayout")
        self.horizontalLayout_2.addLayout(self.horizontalLayout)
        MainWindow.setCentralWidget(self.centralwidget)
        self.menubar = QtWidgets.QMenuBar(MainWindow)
        self.menubar.setGeometry(QtCore.QRect(0, 0, 1014, 30))
        self.menubar.setObjectName("menubar")
        MainWindow.setMenuBar(self.menubar)
        self.statusbar = QtWidgets.QStatusBar(MainWindow)
        self.statusbar.setObjectName("statusbar")
        MainWindow.setStatusBar(self.statusbar)
        self.retranslateUi(MainWindow)
        QtCore.QMetaObject.connectSlotsByName(MainWindow)

    def retranslateUi(self, MainWindow):
        _translate = QtCore.QCoreApplication.translate
        MainWindow.setWindowTitle(_translate("MainWindow", "MainWindow"))


if __name__ == "__main__":
    import sys
    app = QtWidgets.QApplication(sys.argv)
    MainWindow = QtWidgets.QMainWindow()
    ui = Ui_MainWindow()
    ui.setupUi(MainWindow)
    email = 'testemail'
    server = '********'
    tr = QtWidgets.QSystemTrayIcon(QtGui.QIcon(r'payments.PNG'), app)
    print(tr.isSystemTrayAvailable())
    while(1):
        a = extest4.connect(server, email)
        if(a.inbox.unread_count)>0:
            a.inbox.refresh()
            tr.setVisible(True)
            tr.show()
        else:
            tr.hide()
        extest4.close_connections()
        time.sleep(10)
    sys.exit(app.exec_())

您有 3 个阻塞条件:while 周期、邮件检查和 time.sleep。它们都将以某种方式锁定界面(允许在应用程序主事件循环内进行绘画和交互),while 是 "most locking" 一个:直到它退出,GUI 才会被阻止,所以图标没有被绘制(事实上,与 GUI 相关的任何东西都不起作用,包括主要的 window 如果你已经显示它的话)。

由于邮件检查可能需要一些时间来处理,您需要将其放在一个单独的线程中。

在这个例子中,我将逻辑放在 mainwindow 的子类中,因为您可能会将其用作程序的主要 "container"。其他替代方案包括使用 QObject 的子类或 QApplication 本身。

server = 'mail.mailserver.com'
email = 'myname@mailserver.com'

class MailChecker(QtCore.QThread):
    mailAvailable = QtCore.pyqtSignal(bool)
    def __init__(self, server, mail):
        super().__init__()
        self.server = server
        self.mail = mail

    def run(self):
        while True:
            a = extest4.connect(self.server, self.email)
            if a.inbox.unread_count > 0:
                a.inbox.refresh()
                self.mailAvailable.emit(True)
            else:
                self.mailAvailable.emit(False)
            extest4.close_connections()
            QtCore.QThread.sleep(10)


class MainWindow(QtWidgets.QMainWindow, Ui_MainWindow):
    def __init__(self):
        super().__init__()
        self.setupUi(self)
        self.trayIcon = QtWidgets.QSystemTrayIcon(
            QtGui.QIcon(r'payments.PNG'), self)
        self.trayIcon.activated.connect(self.show)
        self.mailChecker = MailChecker(server, email)
        self.mailChecker.mailAvailable.connect(self.trayIcon.setVisible)
        self.mailChecker.start()

if __name__ == "__main__":
    import sys
    app = QtWidgets.QApplication(sys.argv)
    mainWindow = MainWindow()
    sys.exit(app.exec_())