Python Qt GUI设计:窗口之间数据传递(拓展篇—5)

目录

1、单一窗口数据传输

2、多窗口数据传输:调用属性

3、多窗口数据传输:信号与槽


在开发程序时,如果这个程序只有一个窗口,则应该关心这个窗口里面的各个控件之间是如何传递数据的。如果这个程序有多个窗口,那么还应该关心不同的窗口之间是如何传递数据的。

本篇博文首先给出一个例子,说明在一个窗口中不同控件之间的数据是如何传递的。对于多窗口的情况,一般有两种解决方法:一种是主窗口获取子窗口中控件的属性,另一种是通过信号与槽机制,一般是子窗口通过发射信号的形式传递数据,主窗口的槽函数获取这些数据。

1、单一窗口数据传输

对于具有单一窗口的程序来说,一个控件的变化会影响另一个控件的变化,这种变化利用信号与槽机制非常容易解决。

通过示例,了解单一窗口数据传输的方法,效果如下所示:

首先,创建滑块和LCD控件,然后,通过QVBoxLayout设置布局,最后,连接QSlider控件的valueChanged()信号函数和LCD面板控件的display()槽函数。

valueChanged()是QSlider 的一个信号函数,只要 slider 的值发生改变,它就会发射一个信号。还可以设置参数控制信号在什么时候发射,然后通过connect连接信号的接收控件,也就是lcd。槽是对信号的响应,这里是lcd.display,即更新LCD面板的数字信息。

实现代码如下所示:

  1. import sys
  2. from PyQt5.QtWidgets import QWidget,QLCDNumber,QSlider,QVBoxLayout,QApplication
  3. from PyQt5.QtCore import Qt
  4. class WinForm(QWidget):
  5. def __init__(self):
  6. super().__init__()
  7. self.initUI()
  8. def initUI(self):
  9. #1 先创建滑块和 LCD 部件
  10. lcd = QLCDNumber(self)
  11. slider = QSlider(Qt.Horizontal, self)
  12. #2 通过QVboxLayout来设置布局
  13. vBox = QVBoxLayout()
  14. vBox.addWidget(lcd)
  15. vBox.addWidget(slider)
  16. self.setLayout(vBox)
  17. #3 valueChanged()是Qslider的一个信号函数,只要slider的值发生改变,
  18. 它就会发射一个信号,然后通过connect连接信号的接收部件,也就是lcd。
  19. slider.valueChanged.connect(lcd.display)
  20. self.setGeometry(300,300,350,150)
  21. self.setWindowTitle("信号与槽:连接滑块LCD")
  22. if __name__ == '__main__':
  23. app = QApplication(sys.argv)
  24. form = WinForm()
  25. form.show()
  26. sys.exit(app.exec_())

2、多窗口数据传输:调用属性

在PyQt编程过程中,经常会遇到输入或选择多个参数的问题。把多个参数写到一个窗口中,主窗口会显得很臃肿,所以一般是添加一个按钮,调用对话框,在对话框中进行参数的选择,关闭对话框时将参数值返回给主窗口。

PyQt提供了一些标准的对话框类,用于输入数据、修改数据、更改应用的设置等,常见的有QFileDialog、QInputDialog、QColorDialog、QFontDialog等。

本小节将介绍不同窗口在自定义对话框之间通过属性传参。通过示例,了解属性传参的方法,效果如下所示:

示例中,在主窗口中调用对话框有两种方法,单击“弹出对话框"按钮,在对话框的时间日期控件中选择日期,则会把所选中的日期返回到主窗口的lineText文本框控件中。

使用两个按钮(Ok和Cancel)分别连接accept()和reject()槽函数。在类中定义一个静态函数getDateTime(),该静态函数返回3个时间值。原理是利用静态函数的特性,在静态函数中实例化 DateDialog 类,并调用dialog.exec_()函数来显式执行对话框。通过 dialog.exec_()的返回值来判断用户单击的是Ok按钮还是Cancel按钮,然后做出下一步判断。

主窗口实现代码如下所示:

  1. import sys
  2. from PyQt5.QtCore import *
  3. from PyQt5.QtGui import *
  4. from PyQt5.QtWidgets import *
  5. from DateDialog import DateDialog
  6. class WinForm(QWidget):
  7. def __init__(self, parent=None):
  8. super(WinForm, self).__init__(parent)
  9. self.resize(400, 90)
  10. self.setWindowTitle('对话框关闭时返回值给主窗口例子')
  11. self.lineEdit = QLineEdit(self)
  12. self.button1 = QPushButton('弹出对话框1')
  13. self.button1.clicked.connect(self.onButton1Click)
  14. self.button2 = QPushButton('弹出对话框2')
  15. self.button2.clicked.connect(self.onButton2Click)
  16. gridLayout = QGridLayout()
  17. gridLayout.addWidget(self.lineEdit)
  18. gridLayout.addWidget(self.button1)
  19. gridLayout.addWidget(self.button2)
  20. self.setLayout(gridLayout)
  21. def onButton1Click(self):
  22. dialog = DateDialog(self)
  23. result = dialog.exec_()
  24. date = dialog.dateTime()
  25. self.lineEdit.setText(date.date().toString())
  26. print('\n日期对话框的返回值')
  27. print('date=%s' % str(date.date()))
  28. print('time=%s' % str(date.time()))
  29. print('result=%s' % result)
  30. dialog.destroy()
  31. def onButton2Click(self):
  32. date, time, result = DateDialog.getDateTime()
  33. self.lineEdit.setText(date.toString())
  34. print('\n日期对话框的返回值')
  35. print('date=%s' % str(date))
  36. print('time=%s' % str(time))
  37. print('result=%s' % result)
  38. if result == QDialog.Accepted:
  39. print('点击确认按钮')
  40. else:
  41. print('点击取消按钮')
  42. if __name__ == "__main__":
  43. app = QApplication(sys.argv)
  44. form = WinForm()
  45. form.show()
  46. sys.exit(app.exec_())

对话框窗口实现代码如下所示:

  1. from PyQt5.QtCore import *
  2. from PyQt5.QtGui import *
  3. from PyQt5.QtWidgets import *
  4. class DateDialog(QDialog):
  5. def __init__(self, parent=None):
  6. super(DateDialog, self).__init__(parent)
  7. self.setWindowTitle('DateDialog')
  8. # 在布局中添加部件
  9. layout = QVBoxLayout(self)
  10. self.datetime = QDateTimeEdit(self)
  11. self.datetime.setCalendarPopup(True)
  12. self.datetime.setDateTime(QDateTime.currentDateTime())
  13. layout.addWidget(self.datetime)
  14. # 使用两个button(ok和cancel)分别连接accept()和reject()槽函数
  15. buttons = QDialogButtonBox(
  16. QDialogButtonBox.Ok | QDialogButtonBox.Cancel,
  17. Qt.Horizontal, self)
  18. buttons.accepted.connect(self.accept)
  19. buttons.rejected.connect(self.reject)
  20. layout.addWidget(buttons)
  21. # 从对话框中获取当前日期和时间
  22. def dateTime(self):
  23. return self.datetime.dateTime()
  24. # 静态方法创建对话框并返回 (date, time, accepted)
  25. @staticmethod
  26. def getDateTime(parent=None):
  27. dialog = DateDialog(parent)
  28. result = dialog.exec_()
  29. date = dialog.dateTime()
  30. return (date.date(), date.time(), result == QDialog.Accepted)

3、多窗口数据传输:信号与槽

对于多窗口的数据传递,一般是通过子窗口发射信号的,主窗口通过槽函数捕获这个信号,然后获取信号里面的数据。子窗口发射的信号有两种:其中一种是发射PyQt内置的一些信号,另一种是发射自定义的信号。

发射自定义信号的好处是,它的参数类型可以自定义。比如发射一个自定义信号,它的参数类型可以为int、str、dict、list等;如果发射内置信号,则只能是特定的几个参数。

2、多窗口数据传输:调用属性例子基础上进行修改。

主窗口实现代码如下所示:

  1. import sys
  2. from PyQt5.QtCore import *
  3. from PyQt5.QtGui import *
  4. from PyQt5.QtWidgets import *
  5. from DateDialog2 import DateDialog
  6. class WinForm(QWidget):
  7. def __init__(self, parent=None):
  8. super(WinForm, self).__init__(parent)
  9. self.resize(400, 90)
  10. self.setWindowTitle('信号与槽传递参数的示例')
  11. self.open_btn = QPushButton('获取时间')
  12. self.lineEdit_inner = QLineEdit(self)
  13. self.lineEdit_emit = QLineEdit(self)
  14. self.open_btn.clicked.connect(self.openDialog)
  15. self.lineEdit_inner.setText('接收子窗口内置信号的时间')
  16. self.lineEdit_emit.setText('接收子窗口自定义信号的时间')
  17. grid = QGridLayout()
  18. grid.addWidget(self.lineEdit_inner)
  19. grid.addWidget(self.lineEdit_emit)
  20. grid.addWidget(self.open_btn)
  21. self.setLayout(grid)
  22. def openDialog(self):
  23. dialog = DateDialog(self)
  24. '''连接子窗口的内置信号与主窗口的槽函数'''
  25. dialog.datetime_inner.dateTimeChanged.connect(self.deal_inner_slot)
  26. '''连接子窗口的自定义信号与主窗口的槽函数'''
  27. dialog.Signal_OneParameter.connect(self.deal_emit_slot)
  28. dialog.show()
  29. def deal_inner_slot(self, date):
  30. self.lineEdit_inner.setText(date.toString())
  31. def deal_emit_slot(self, dateStr):
  32. self.lineEdit_emit.setText(dateStr)
  33. if __name__ == "__main__":
  34. app = QApplication(sys.argv)
  35. form = WinForm()
  36. form.show()
  37. sys.exit(app.exec_())

对话框窗口实现代码如下所示:

  1. from PyQt5.QtCore import *
  2. from PyQt5.QtGui import *
  3. from PyQt5.QtWidgets import *
  4. class DateDialog(QDialog):
  5. Signal_OneParameter = pyqtSignal(str)
  6. def __init__(self, parent=None):
  7. super(DateDialog, self).__init__(parent)
  8. self.setWindowTitle('子窗口:用来发射信号')
  9. # 在布局中添加部件
  10. layout = QVBoxLayout(self)
  11. self.label = QLabel(self)
  12. self.label.setText('前者发射内置信号\n后者发射自定义信号')
  13. self.datetime_inner = QDateTimeEdit(self)
  14. self.datetime_inner.setCalendarPopup(True)
  15. self.datetime_inner.setDateTime(QDateTime.currentDateTime())
  16. self.datetime_emit = QDateTimeEdit(self)
  17. self.datetime_emit.setCalendarPopup(True)
  18. self.datetime_emit.setDateTime(QDateTime.currentDateTime())
  19. layout.addWidget(self.label)
  20. layout.addWidget(self.datetime_inner)
  21. layout.addWidget(self.datetime_emit)
  22. # 使用两个button(ok和cancel)分别连接accept()和reject()槽函数
  23. buttons = QDialogButtonBox(
  24. QDialogButtonBox.Ok | QDialogButtonBox.Cancel,
  25. Qt.Horizontal, self)
  26. buttons.accepted.connect(self.accept)
  27. buttons.rejected.connect(self.reject)
  28. layout.addWidget(buttons)
  29. self.datetime_emit.dateTimeChanged.connect(self.emit_signal)
  30. def emit_signal(self):
  31. date_str = self.datetime_emit.dateTime().toString()
  32. self.Signal_OneParameter.emit(date_str)

关注公众号,发送关键字:Java车牌识别,获取项目源码。