如何在 Qlabel 中显示来自外部调用的 Python 函数的消息
How to display message in Qlabel from externally called Python function
我实际上有 3 个问题。
我有一个调用外部函数的按钮(下面给出的代码示例)。当我按下 运行 按钮时,功能正在执行。我想在 window 中将来自该函数的消息显示到我的 Qlabel。到目前为止,我在函数完全执行时显示该消息,并且 returns 到主函数。但我想在函数执行过程中看到这一点。有办法解决吗?
其次,我的GUI在执行任何外部函数时进入'not responding'模式。这可能是什么原因以及解决方法?
我想在以任何其他分辨率(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和问题2是相关的,所以我将它们一起解决:
GUI 冻结是因为睡眠函数阻塞了 GUI 事件循环,所以逻辑是 运行 它在另一个线程中。另一方面,GUI 不能直接从另一个线程访问,因为它不是线程安全的,因此您必须使用信号。
main.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_())
callingexternalfunction.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>
我实际上有 3 个问题。
我有一个调用外部函数的按钮(下面给出的代码示例)。当我按下 运行 按钮时,功能正在执行。我想在 window 中将来自该函数的消息显示到我的 Qlabel。到目前为止,我在函数完全执行时显示该消息,并且 returns 到主函数。但我想在函数执行过程中看到这一点。有办法解决吗?
其次,我的GUI在执行任何外部函数时进入'not responding'模式。这可能是什么原因以及解决方法?
我想在以任何其他分辨率(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和问题2是相关的,所以我将它们一起解决:
GUI 冻结是因为睡眠函数阻塞了 GUI 事件循环,所以逻辑是 运行 它在另一个线程中。另一方面,GUI 不能直接从另一个线程访问,因为它不是线程安全的,因此您必须使用信号。
main.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_())
callingexternalfunction.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>