0

当启动某些操作时,我想在我的 GUI 中添加一个 throbber。

这是我的脚本:

class StartTask(QtCore.QThread):
    taskStarted = pyqtSignal()
    def run(self):
        self.taskStarted.emit()

class StopTask(QtCore.QThread):
    taskStopped = pyqtSignal()
    def run(self):
        self.taskStopped.emit()

class Projet(object):

    def __init__(self):
        self.movie = '' # throbber
        self.startTask = StartTask()
        self.startTask.taskStarted.connect(self.startThrobber)
        self.stopTask = StopTask()
        self.stopTask.taskStopped.connect(self.stopThrobber)

    def startThrobber(self):
        # set up the movie screen on a label
        self.movie_screen = QLabel()
        # expand and center the label
        main_layout = QVBoxLayout()
        main_layout.addWidget(self.movie_screen)
        ui.throbberTab2.setLayout(main_layout)
        # use an animated gif file you have in the working folder
        byteF = QByteArray()
        movie = QMovie("D:\Various\Images\loader.gif", byteF)
        movie.setCacheMode(QMovie.CacheAll)
        movie.setSpeed(100)
        self.movie_screen.setMovie(movie)
        movie.start()

        return movie

    def stopThrobber(self):
        movie1 = self.startThrobber()
        movie1.stop()

    def goAction(self):
        if ui.chkbox.isChecked():
            self.startTask.taskStarted.connect(self.startThrobber)
            os.system(r'..........') # script launched
            self.stopTask.taskStopped.connect(self.stopThrobber)
            QMessageBox.information(self.popup(), "Information", "It works!")

因为这是我第一次使用线程,所以我找不到哪里出了问题,哪里出了问题..

这没有结果,即使我认为我离正确的代码并不太远。

我已经设法让颤动出现,但不是在正确的时刻(当时线程没有工作)。

4

2 回答 2

2

您的代码存在一个问题,即调用os.system()(可能是长时间运行,因此需要一个 throbber)正在主线程中执行。这会阻止 GUI(和 Qt 事件循环)。QMovie需要一个正常运行的事件循环才能正确设置动画。因此,对于您当前的代码,您无法使颤动显示为动画,因为这需要响应式 GUI(或更准确地说是响应式事件循环)。

顺便说一句,您也不能直接从线程调用 GUI 方法。您只能从辅助线程将事件发送回主线程(就像您当前所做的那样)。这意味着您不能将其卸载QMovie到线程。

因此,您需要将调用卸载os.system()到线程中,并在完成时发出信号。该信号可以连接到停止颤动的插槽。在启动线程之前,只需直接调用您现有的startThrobber()方法来启动 throbber。

这样,您的 UI 保持响应(包括正确显示颤动动画),同时线程处理阻塞所有内容的长时间运行的进程。

于 2015-06-01T08:37:27.100 回答
0

我没有使用 a QThread,而是使用了subprocesswhich 工作正常。不过,我不知道其中的区别,例如一个是否比另一个“更好”,或者它们是否存在细微差异等,但它确实有效!

于 2015-06-01T10:48:05.247 回答