如何从外部调用的 Python 函数在 Qlabel 中显示消息

Posted

技术标签:

【中文标题】如何从外部调用的 Python 函数在 Qlabel 中显示消息【英文标题】:How to display message in Qlabel from externally called Python function 【发布时间】:2020-03-24 21:16:30 【问题描述】:

我实际上有 3 个问题。

    我有一个调用外部函数的按钮(代码示例如下)。当我按下运行按钮时,功能正在执行。我想在窗口中显示来自该函数的消息到我的 Qlabel。到目前为止,当函数完全执行并返回主函数时,我会显示该消息。但我想看看在函数执行过程中。有办法解决吗?

    其次,我的 GUI 在执行任何外部功能时会进入“无响应”模式。这可能是什么原因以及解决方法?

    当我以任何其他分辨率 (ScreenSize) 打开 GUI 时,我想动态调整所有小部件的大小。现在,无论如何 QTablewidget 的大小都是固定的。我怎么才能得到它?这可以通过 QTDesigner 中的任何设置来实现吗?

UI 设计师生成的代码:

from PyQt5 import QtCore, QtGui, QtWidgets


class Ui_MainWindow(object):
    def setupUi(self, MainWindow):
        MainWindow.setObjectName("MainWindow")
        MainWindow.resize(1150, 905)
        self.centralwidget = QtWidgets.QWidget(MainWindow)
        self.centralwidget.setObjectName("centralwidget")
        self.pushButton = QtWidgets.QPushButton(self.centralwidget)
        self.pushButton.setGeometry(QtCore.QRect(290, 820, 161, 41))
        self.pushButton.setObjectName("pushButton")
        self.tableWidget = QtWidgets.QTableWidget(self.centralwidget)
        self.tableWidget.setGeometry(QtCore.QRect(10, 10, 1131, 791))
        self.tableWidget.setObjectName("tableWidget")
        self.tableWidget.setColumnCount(0)
        self.tableWidget.setRowCount(0)
        self.label = QtWidgets.QLabel(self.centralwidget)
        self.label.setGeometry(QtCore.QRect(580, 820, 191, 41))
        self.label.setText("")
        self.label.setObjectName("label")
        MainWindow.setCentralWidget(self.centralwidget)
        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"))
        self.pushButton.setText(_translate("MainWindow", "Run"))


if __name__ == "__main__":
    import sys
    app = QtWidgets.QApplication(sys.argv)
    MainWindow = QtWidgets.QMainWindow()
    ui = Ui_MainWindow()
    ui.setupUi(MainWindow)
    MainWindow.show()
    sys.exit(app.exec_())

主脚本:

from PyQt5 import QtWidgets

from demoUI import Ui_MainWindow

from callingexternalfunction import callingexternalfunction

class DemoCode(QtWidgets.QMainWindow, Ui_MainWindow):                    
    def __init__(self):
        super(DemoCode, self).__init__()
        self.setupUi(self)

        self.pushButton.clicked.connect(self.button_clicked)

    def button_clicked(self):
        callingexternalfunction(self)

if __name__ == "__main__":
    import sys
    app = QtWidgets.QApplication(sys.argv)
    window = DemoCode()    
    window.show()
    sys.exit(app.exec_())

函数调用外部函数:

import time

def callingexternalfunction(self):
    for i in range(0,5):
        time.sleep(3)
        self.label.setText('3 Second Pause '+ str(i))

【问题讨论】:

【参考方案1】:

问题 1 和问题 2 相关,所以我将一起解决:

GUI 冻结是因为 sleep 函数阻塞了 GUI 事件循环,因此逻辑是在另一个线程中运行它。另一方面,不能从另一个线程直接访问 GUI,因为它不是线程安全的,所以您必须使用信号。

ma​​in.py

from PyQt5 import QtCore, QtWidgets

import threading

from demoUI import Ui_MainWindow

from callingexternalfunction import callingexternalfunction


class DemoCode(QtWidgets.QMainWindow, Ui_MainWindow):
    signal = QtCore.pyqtSignal(str)

    def __init__(self):
        super(DemoCode, self).__init__()
        self.setupUi(self)

        self.pushButton.clicked.connect(self.button_clicked)
        self.signal.connect(self.label.setText)

    def button_clicked(self):
        threading.Thread(
            target=callingexternalfunction, args=(self.signal,), daemon=True
        ).start()


if __name__ == "__main__":
    import sys

    app = QtWidgets.QApplication(sys.argv)
    window = DemoCode()
    window.show()
    sys.exit(app.exec_())

调用externalfunction.py

import time

def callingexternalfunction(signal):
    for i in range(0,5):
        time.sleep(3)
        signal.emit('3 Second Pause '+ str(i))

必须使用布局来调整小部件的大小(Here 是对其进行解释的官方教程),在这种情况下,您可以使用 QGridLayout 以及其他修改:

<?xml version="1.0" encoding="UTF-8"?>
<ui version="4.0">
 <class>MainWindow</class>
 <widget class="QMainWindow" name="MainWindow">
  <property name="geometry">
   <rect>
    <x>0</x>
    <y>0</y>
    <width>800</width>
    <height>600</height>
   </rect>
  </property>
  <property name="windowTitle">
   <string>MainWindow</string>
  </property>
  <widget class="QWidget" name="centralwidget">
   <layout class="QGridLayout" name="gridLayout" columnstretch="0,1">
    <item row="0" column="0" colspan="2">
     <widget class="QTableWidget" name="tableWidget"/>
    </item>
    <item row="1" column="0">
     <widget class="QPushButton" name="pushButton">
      <property name="minimumSize">
       <size>
        <width>160</width>
        <height>40</height>
       </size>
      </property>
      <property name="baseSize">
       <size>
        <width>0</width>
        <height>0</height>
       </size>
      </property>
      <property name="text">
       <string>Run</string>
      </property>
     </widget>
    </item>
    <item row="1" column="1">
     <widget class="QLabel" name="label">
      <property name="text">
       <string/>
      </property>
      <property name="margin">
       <number>0</number>
      </property>
      <property name="indent">
       <number>40</number>
      </property>
     </widget>
    </item>
   </layout>
  </widget>
  <widget class="QMenuBar" name="menubar">
   <property name="geometry">
    <rect>
     <x>0</x>
     <y>0</y>
     <width>800</width>
     <height>26</height>
    </rect>
   </property>
  </widget>
  <widget class="QStatusBar" name="statusbar"/>
 </widget>
 <resources/>
 <connections/>
</ui>

【讨论】:

谢谢!它按预期工作!关于调整小部件的大小,我尝试使用布局,但我拥有的 GUI 有点复杂。还有其他解决方法吗?我想在布局中使用组框。实施起来难吗?

以上是关于如何从外部调用的 Python 函数在 Qlabel 中显示消息的主要内容,如果未能解决你的问题,请参考以下文章

如何从外部调用 $(document).ready 中的函数

如何从外部控制器函数调用控制器函数,如 sencha touch 中的全局函数

如何使用从外部脚本调用的回调发送信号?

如何从外部 JavaScript 文件调用反应函数

如何使用从外部脚本调用的回调发送信号?

从iframe外部调用iframe内部的函数[重复]