我需要为一个类创建一个UI,该类通过一组变量进行大量计算。代码大致如下所示:
class MainWindow:
def __init__(self):
pulse = Pulse()
self.var1_lineEdit.textEdited.connect(self.set_attribute(pulse, "var1", new_value))
self.var2_lineEdit.textEdited.connect(self.set_attribute(pulse, "var2", new_value))
self.var3_lineEdit.textEdited.connect(self.set_attribute(pulse, "var3", new_value))
....
def set_attribute(pulse, var_name, value):
# get value from line edit
# update value in pulse
# update other calculations
class Pulse:
def __init__(self):
var1 = 1
var2 = 2
var3 = 3
....
我的问题是如何编写set_attribute代码。有没有办法通过将名称作为函数参数传递来访问另一个类的各个变量?我想避免为每个变量定义一个新函数。
或者有更简单的方法吗?我不太熟悉OOP和编写UI的正确方法。
答案 0 :(得分:1)
修改强>
您提出的信号/插槽连接会让您遇到麻烦。使用connect函数将信号连接到插槽,该函数接受python callable作为其参数 - 而不是函数调用本身。这是
之间的区别widget.textChanged.connect(self.set_attribute) # right syntax
widget.textChanged.connect(self.set_attribute()) # wrong syntax
那么,你如何告诉set_attribute
该做什么? ekhumoro告诉我我的方式错误:使用QSignalMapper
。这是一种过时的技术。最好只使用lambda
函数(或partials
)来获得所需的效果。
我更新了我的演示以使用lambdas
出于本演示的目的,我创建了一个简单的QMainWindow
,其中包含两个QLineEdit
小部件和一个QPushButton
。
# file ui_main.py
from PyQt5 import QtCore, QtGui, QtWidgets
class Ui_MainWindow(object):
def setupUi(self, MainWindow):
MainWindow.setObjectName("MainWindow")
MainWindow.resize(359, 249)
self.centralwidget = QtWidgets.QWidget(MainWindow)
self.centralwidget.setObjectName("centralwidget")
self.formLayoutWidget = QtWidgets.QWidget(self.centralwidget)
self.formLayoutWidget.setGeometry(QtCore.QRect(0, 0, 351, 51))
self.formLayoutWidget.setObjectName("formLayoutWidget")
self.formLayout = QtWidgets.QFormLayout(self.formLayoutWidget)
self.formLayout.setObjectName("formLayout")
self.configParam1Label = QtWidgets.QLabel(self.formLayoutWidget)
self.configParam1Label.setObjectName("configParam1Label")
self.formLayout.setWidget(0, QtWidgets.QFormLayout.LabelRole, self.configParam1Label)
self.configEntry = QtWidgets.QLineEdit(self.formLayoutWidget)
self.configEntry.setObjectName("configEntry")
self.formLayout.setWidget(0, QtWidgets.QFormLayout.FieldRole, self.configEntry)
self.configParam2Label = QtWidgets.QLabel(self.formLayoutWidget)
self.configParam2Label.setObjectName("configParam2Label")
self.formLayout.setWidget(1, QtWidgets.QFormLayout.LabelRole, self.configParam2Label)
self.configEntry2 = QtWidgets.QLineEdit(self.formLayoutWidget)
self.configEntry2.setObjectName("configEntry2")
self.formLayout.setWidget(1, QtWidgets.QFormLayout.FieldRole, self.configEntry2)
self.pulseButton = QtWidgets.QPushButton(self.centralwidget)
self.pulseButton.setGeometry(QtCore.QRect(130, 140, 75, 23))
self.pulseButton.setObjectName("pulseButton")
MainWindow.setCentralWidget(self.centralwidget)
self.retranslateUi(MainWindow)
QtCore.QMetaObject.connectSlotsByName(MainWindow)
def retranslateUi(self, MainWindow):
_translate = QtCore.QCoreApplication.translate
MainWindow.setWindowTitle(_translate("MainWindow", "MainWindow"))
self.configParam1Label.setText(_translate("MainWindow", "Config Param 1"))
self.configParam2Label.setText(_translate("MainWindow", "Config Param 2"))
self.pulseButton.setText(_translate("MainWindow", "GetPulse"))
在另一个文件中,在一些导入之后,定义Pulse
- 一个包含两个属性的简单类
# file main.py
from PyQt5.QtCore import (QSignalMapper, pyqtSlot)
from PyQt5.QtWidgets import (QApplication, QMainWindow, QMessageBox)
from ui_main import Ui_MainWindow
class Pulse:
def __init__(self):
self.param_one = None
self.param_two = None
现在,定义MainWindow
,设置其用户界面并为其提供Pulse
:
class MainWindow(QMainWindow):
def __init__(self, parent=None):
super(MainWindow, self).__init__(parent)
self.pulse = Pulse()
self.ui = Ui_MainWindow()
self.ui.setupUi(self)
self.widget_one = self.ui.configEntry
self.widget_two = self.ui.configEntry2
self.pulse_button = self.ui.pulseButton
self.widget_one.textChanged.connect(lambda: self.widget_handler('param_one', self.widget_one.text()))
self.widget_one.textChanged.connect(lambda: self.widget_handler('param_two', self.widget_two.text()))
def widget_handler(self, attr, value):
setattr(self.pulse,attr,value)
def handler(self, idx):
widget, attr = self.attribute_widget_list[idx]
widget_value = widget.text()
setattr(self.pulse,attr,widget_value)
setattr
相当于,例如self.pulse.param_one = widget_value
。
为了证明它有效,我们使用QPushButton
来探测Pulse
的状态:
@pyqtSlot()
def on_pulseButton_clicked(self):
message = QMessageBox()
message.setText("{} - {}".format(self.pulse.param_one, self.pulse.param_two))
message.exec()
这里我使用的是不同的信号/插槽连接方案,您可以阅读here
以下是我们运行脚本的方式:
if __name__ == '__main__':
import sys
app = QApplication(sys.argv)
window = MainWindow()
window.show()
sys.exit(app.exec_())
希望有所帮助!