首页 > 解决方案 > PyQT5 qsytemtrayicon

问题描述

我对 qsystemtray 有疑问。当我在没有“while”的情况下运行此代码时,我可以看到图标。用while I can't watch icon(我检查了程序进入条件“if (a.inbox.unread_count)> 0”),但是icon的区域一定是不可用的。也就是出现了托盘图标,但是是未绘制。有什么方法可以解决这个问题吗?操作系统: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_())

标签: pythonpyqt5qsystemtrayicon

解决方案


您有 3 个阻塞条件:while循环、邮件检查和 time.sleep。所有这些都会以某种方式锁定界面(允许在应用程序主事件循环中进行绘制和交互),其中while“锁定最多”的方式是:在退出之前,GUI 被阻塞,因此图标不会被绘制(事实上,与 GUI 相关的任何内容都不起作用,包括主窗口(如果您已显示它)。

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

在本例中,我将逻辑放在主窗口的子类中,因为您可能会将其用作程序的主要“容器”。其他替代方法包括使用 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_())

推荐阅读