qsplashscreen не отображается должным образом, пока app.processEvents() не вызывается дважды
PyQT4.11
Python 2.7
Я пытался создать заставку для отображения сообщения о загрузке, когда выбрана конкретная вкладка. когда app.processEvents()
называется, я получаю серое поле отображается вместо изображения, но если я добавлю второй app.processEvents()
после вызова первой функции отображается правильное изображение, пока всплеск не будет закрыт по завершении.
Я играл с таймерами сна, перекрашиванием, обновлением, закрытием и повторным отображением вкладки и вкладки со смешанными результатами, в редких случаях это будет работать, но чаще всего не будет. Кто-нибудь может пролить свет на то, что мне не хватает?
import sys
from PyQt4 import QtCore, QtGui
class splashTest(QtGui.QDialog):
def __init__(self, parent=None):
super(splashTest, self).__init__(parent)
self.setGeometry(000,000,800,400)
self.tabWidget = QtGui.QTabWidget(self)
self.tabWidget.setGeometry(QtCore.QRect(0, 0, 500, 500))
self.tabWidget.setObjectName("tabWidget")
self.tab1 = QtGui.QWidget()
self.tab2 = QtGui.QWidget()
self.tab3 = QtGui.QWidget()
self.tabWidget.addTab(self.tab1, "tab1")
self.tabWidget.addTab(self.tab2, "tab2")
self.tabWidget.addTab(self.tab3, "Click Me")
self.tabWidget.setCurrentIndex(1)
self.tabWidget.currentChanged.connect(self.whichTabIsSelected)
def whichTabIsSelected(self):
if self.tabWidget.currentIndex() == 2:
splash_pix = QtGui.QPixmap('splash.png')
splash = QtGui.QSplashScreen(splash_pix, QtCore.Qt.WindowStaysOnTopHint)
splash.setMask(splash_pix.mask())
splash.show()
app.processEvents()
self.timeConsumingFunction()
app.processEvents()
self.timeConsumingFunction()
splash.finish(self.tabWidget)
else:
pass
def timeConsumingFunction(self):
b = 0
for a in range(100000000):
b += a
print (b)
if __name__ == "__main__":
app = QtGui.QApplication(sys.argv)
myapp = splashTest()
myapp.show()
sys.exit(app.exec_())
1 ответ
Проблема в том, что вы выполняете тяжелую задачу в главном потоке, блокирующем цикл событий графического интерфейса, и это приводит к зависанию окна, и вы принудительно обновляете QApplication::processEvents()
вместо этого правильным решением является выполнение этой задачи в другом потоке.
import sys
from PyQt4 import QtCore, QtGui
import threading
class splashTest(QtGui.QDialog):
finished = QtCore.pyqtSignal()
def __init__(self, parent=None):
super(splashTest, self).__init__(parent)
self.setGeometry(000,000,800,400)
self.tabWidget = QtGui.QTabWidget(self)
self.tabWidget.setGeometry(QtCore.QRect(0, 0, 500, 500))
self.tabWidget.setObjectName("tabWidget")
self.tab1 = QtGui.QWidget()
self.tab2 = QtGui.QWidget()
self.tab3 = QtGui.QWidget()
self.tabWidget.addTab(self.tab1, "tab1")
self.tabWidget.addTab(self.tab2, "tab2")
self.tabWidget.addTab(self.tab3, "Click Me")
self.tabWidget.setCurrentIndex(1)
self.tabWidget.currentChanged.connect(self.whichTabIsSelected)
@QtCore.pyqtSlot(int)
def whichTabIsSelected(self, index):
if index == 2:
splash_pix = QtGui.QPixmap('splash.png')
splash = QtGui.QSplashScreen(self, splash_pix, QtCore.Qt.WindowStaysOnTopHint)
splash.setMask(splash_pix.mask())
splash.show()
for _ in range(2):
loop = QtCore.QEventLoop()
self.finished.connect(loop.quit)
threading.Thread(target=self.timeConsumingFunction).start()
loop.exec_()
splash.finish(self.tabWidget)
splash.close()
def timeConsumingFunction(self):
b = 0
for a in range(100000000):
b += a
print (b)
self.finished.emit()
if __name__ == "__main__":
app = QtGui.QApplication(sys.argv)
myapp = splashTest()
myapp.show()
sys.exit(app.exec_())