python - 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_())
解决方案
您有 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_())
推荐阅读
- javascript - 为什么0的无穷大被实现为1?
- javascript - Javascript自动完成当前行并添加{}花括号?
- javascript - 如何在反应中重置不是表单字段的输入字段
- vue.js - VueJS中的Http请求后在运行时渲染组件
- c - Unix 域 SOCK_DGRAM 和 SOCK_SEQPACKET 之间的区别?
- database - 如何在没有桌面界面的ubuntu 18.04服务器上安装oracle 11g?
- swift - 在 Swift 中使用 GKGraph.findpath() 方法显然是随机执行时间
- java - Java:对象数组中的对象数组
- macos - MacOS - SwiftUI,选择器关闭弹出框
- google-cloud-firestore - Firestore 获取子集合字段 == 'value' 的所有父文档