PyQt5 加载旋转器根据发布请求停止

Ste*_*fan 3 python pyqt pyqt4 python-requests pyqt5

我正在使用在这里找到的 QtWaitingSpinner: https: //github.com/snowwlex/QtWaitingSpinner。您可以像这样创建并启动微调器spinner = QtWaitingSpinner(self); spinner.start():不幸的是,当我尝试从 GUI 发出 POST 请求时,旋转器会停止,直到返回响应为止。因此,我根本看不到旋转器,或者如果我过早启动旋转器,它会在等待响应时停止旋转。我想我必须使用某种异步方法,如 QThread 或 asyncio,但目前还不清楚解决这个问题的最佳方法是什么。如果有人能告诉我最好的处理方法,我将不胜感激。这是我正在做的事情的简化版本:

class Obj(QDialog):
    # some button calls this function when pressed
    def submit(self):
        #start spinner
        spinner = QtWaitingSpinner(self)
        spinner.start()

        # post some data to some url, spinner should spin
        r = requests.post('some_url.com', json=some_data)

        # stop spinner
        spinner.stop()
Run Code Online (Sandbox Code Playgroud)

eyl*_*esc 6

您遇到的问题requests是阻塞 Qt 循环,因此像这样的元素QTimer不起作用。一种解决方案是在另一个线程上运行该任务,一种简单的方法是使用QRunnableQThreadPool

class RequestRunnable(QRunnable):
    def __init__(self, url, json, dialog):
        QRunnable.__init__(self)
        self.mUrl = url
        self.mJson = json
        self.w = dialog

    def run(self):
        r = requests.post(self.mUrl, json=self.mJson)
        QMetaObject.invokeMethod(self.w, "setData",
                                 Qt.QueuedConnection,
                                 Q_ARG(str, r.text))


class Dialog(QDialog):
    def __init__(self, *args, **kwargs):
        QDialog.__init__(self, *args, **kwargs)
        self.setLayout(QVBoxLayout())
        btn = QPushButton("Submit", self)
        btn.clicked.connect(self.submit)
        self.spinner = QtWaitingSpinner(self)

        self.layout().addWidget(btn)
        self.layout().addWidget(self.spinner)

    def submit(self):
        self.spinner.start()
        runnable = RequestRunnable("https://api.github.com/some/endpoint",
                                   {'some': 'data'},
                                   self)
        QThreadPool.globalInstance().start(runnable)

    @pyqtSlot(str)
    def setData(self, data):
        print(data)
        self.spinner.stop()
        self.adjustSize()
Run Code Online (Sandbox Code Playgroud)

完整的示例可以在以下链接中找到