Обновление PyQt GUI из потока Python - PullRequest
0 голосов
/ 30 апреля 2018

У меня есть GUI, созданный в Designer (pyqt5). Функция в моем основном классе должна работать в отдельном потоке. Я также ловлю стандартный вывод QtextEdit LIVE во время операций. Пока все работает.

Прямо сейчас я пытаюсь внедрить ProgressBar в мою основную форму графического интерфейса. Панель должна отображать прогрессию в реальном времени так же, как в textEdit.

Пример кода ниже работает на Linux без каких-либо предупреждений. Но в Windows я получаю сообщение об ошибке:

QObject::setParent: Cannot set parent, new parent is in a different thread

Я знаю, что это из-за того, что я изменил элемент пользовательского интерфейса в своей поточной функции. Я провел свое исследование, но все ответы указывают на использование QThreads (именно тогда, когда я начал понимать основы многопоточности!). Я бы предпочел способ обновления моего графического интерфейса без необходимости изменять текущую систему потоков ниже.

Вот пример кода:

import sys
import threading
import time

from PyQt5.QtWidgets import QApplication, QMainWindow
from PyQt5.QtCore import QObject, pyqtSignal
from PyQt5.QtGui import QTextCursor

from ui_form import Ui_Form


class EmittingStream(QObject):
    textWritten = pyqtSignal(str)

    def write(self, text):
        self.textWritten.emit(str(text))


class Form(QMainWindow):
    finished = pyqtSignal()

    def __init__(self, parent=None):
        super(Form, self).__init__(parent)

        # Install the custom output stream
        sys.stdout = EmittingStream(textWritten=self.normalOutputWritten)

        self.ui = Ui_Form()
        self.ui.setupUi(self)

        self.ui.pushButton_run.clicked.connect(self.start_task)
        self.finished.connect(self.end_task)

    def start_task(self):

        self.thread = threading.Thread(target=self.run_test)
        self.thread.start()
        self.ui.pushButton_run.setEnabled(False)

    def end_task(self):
        self.ui.pushButton_run.setEnabled(True)

    def __del__(self):
        # Restore sys.stdout
        sys.stdout = sys.__stdout__

    def normalOutputWritten(self, text):
        """Append text to the QTextEdit."""
        cursor = self.ui.textEdit.textCursor()
        cursor.movePosition(QTextCursor.End)
        cursor.insertText(text)
        self.ui.textEdit.setTextCursor(cursor)
        self.ui.textEdit.ensureCursorVisible()

    def run_test(self):
        for i in range(100):
            per = i + 1
            self.ui.progressBar.setValue(per)
            print("%%%s" % per)
            time.sleep(0.15)  # simulating expensive task

        print("Task Completed!")
        time.sleep(1.5)
        self.ui.progressBar.reset()
        self.finished.emit()


def main():
    app = QApplication(sys.argv)
    form = Form()
    form.show()
    sys.exit(app.exec_())


if __name__ == "__main__":
    main()

пользовательский интерфейс:

# -*- coding: utf-8 -*-
# Form implementation generated from reading ui file 'form.ui'
#
# Created: Mon Apr 30 13:43:19 2018
#      by: PyQt5 UI code generator 5.2.1
#
# WARNING! All changes made in this file will be lost!

from PyQt5 import QtCore, QtGui, QtWidgets


class Ui_Form(object):
    def setupUi(self, Form):
        Form.setObjectName("Form")
        Form.resize(800, 600)
        self.centralwidget = QtWidgets.QWidget(Form)
        self.centralwidget.setObjectName("centralwidget")
        self.pushButton_run = QtWidgets.QPushButton(self.centralwidget)
        self.pushButton_run.setGeometry(QtCore.QRect(40, 20, 311, 191))
        self.pushButton_run.setObjectName("pushButton_run")
        self.textEdit = QtWidgets.QTextEdit(self.centralwidget)
        self.textEdit.setGeometry(QtCore.QRect(40, 230, 721, 241))
        self.textEdit.setObjectName("textEdit")
        self.progressBar = QtWidgets.QProgressBar(self.centralwidget)
        self.progressBar.setGeometry(QtCore.QRect(40, 490, 721, 23))
        self.progressBar.setObjectName("progressBar")
        Form.setCentralWidget(self.centralwidget)
        self.menubar = QtWidgets.QMenuBar(Form)
        self.menubar.setGeometry(QtCore.QRect(0, 0, 800, 25))
        self.menubar.setObjectName("menubar")
        Form.setMenuBar(self.menubar)
        self.statusbar = QtWidgets.QStatusBar(Form)
        self.statusbar.setObjectName("statusbar")
        Form.setStatusBar(self.statusbar)

        self.retranslateUi(Form)
        QtCore.QMetaObject.connectSlotsByName(Form)

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

Каким-то образом мне нужно - постоянно - сообщить потоку графического интерфейса (из моего рабочего потока), что значение индикатора выполнения меняется (процесс, который может занять несколько минут).

1 Ответ

0 голосов
/ 30 апреля 2018

Определить пользовательский сигнал, который отправляет обновления на индикатор выполнения:

class Form(QMainWindow):
    finished = pyqtSignal()
    updateProgress = pyqtSignal(int)

    def __init__(self, parent=None):
        super(Form, self).__init__(parent)
        ...
        self.updateProgress.connect(self.ui.progressBar.setValue)

    def run_test(self):
        for i in range(100):
            per = i + 1
            self.updateProgress.emit(per)
            ...
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...