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

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

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

1、单一窗口数据传输

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

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

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

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

实现代码如下所示:

import sys

from PyQt5.QtWidgets import QWidget,QLCDNumber,QSlider,QVBoxLayout,QApplication

from PyQt5.QtCore import Qt

class WinForm(QWidget):

    def __init__(self):

        super().__init__() 

        self.initUI()

    def initUI(self):

        #1 先创建滑块和 LCD 部件

        lcd = QLCDNumber(self)

        slider = QSlider(Qt.Horizontal, self)

        #2 通过QVboxLayout来设置布局

        vBox = QVBoxLayout()

        vBox.addWidget(lcd)

        vBox.addWidget(slider)

        self.setLayout(vBox)

        #3 valueChanged()是Qslider的一个信号函数,只要slider的值发生改变,它就会发射一个信号,然后通过connect连接信号的接收部件,也就是lcd。

        slider.valueChanged.connect(lcd.display)

        self.setGeometry(300,300,350,150)

        self.setWindowTitle("信号与槽:连接滑块LCD")

if __name__ == '__main__':

    app = QApplication(sys.argv)

    form = WinForm()

    form.show()                     

    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按钮,然后做出下一步判断。

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

import sys

from PyQt5.QtCore import *

from PyQt5.QtGui import *

from PyQt5.QtWidgets import *

from DateDialog import DateDialog

class WinForm(QWidget):

    def __init__(self, parent=None):

        super(WinForm, self).__init__(parent)

        self.resize(400, 90)

        self.setWindowTitle('对话框关闭时返回值给主窗口例子')

        self.lineEdit = QLineEdit(self)

        self.button1 = QPushButton('弹出对话框1')

        self.button1.clicked.connect(self.onButton1Click)

        self.button2 = QPushButton('弹出对话框2')

        self.button2.clicked.connect(self.onButton2Click)

        gridLayout = QGridLayout()

        gridLayout.addWidget(self.lineEdit)

        gridLayout.addWidget(self.button1)

        gridLayout.addWidget(self.button2)

        self.setLayout(gridLayout)

    def onButton1Click(self):

        dialog = DateDialog(self)

        result = dialog.exec_()

        date = dialog.dateTime()

        self.lineEdit.setText(date.date().toString())

        print('\n日期对话框的返回值')

        print('date=%s' % str(date.date()))

        print('time=%s' % str(date.time()))

        print('result=%s' % result)

        dialog.destroy()

    def onButton2Click(self):

        date, time, result = DateDialog.getDateTime()

        self.lineEdit.setText(date.toString())

        print('\n日期对话框的返回值')

        print('date=%s' % str(date))

        print('time=%s' % str(time))

        print('result=%s' % result)

        if result == QDialog.Accepted:

            print('点击确认按钮')

        else:

            print('点击取消按钮')

if __name__ == "__main__":

    app = QApplication(sys.argv)

    form = WinForm()

    form.show()

    sys.exit(app.exec_())

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

from PyQt5.QtCore import *

from PyQt5.QtGui import *

from PyQt5.QtWidgets import *

class DateDialog(QDialog):

    def __init__(self, parent=None):

        super(DateDialog, self).__init__(parent)

        self.setWindowTitle('DateDialog')

        # 在布局中添加部件

        layout = QVBoxLayout(self)

        self.datetime = QDateTimeEdit(self)

        self.datetime.setCalendarPopup(True)

        self.datetime.setDateTime(QDateTime.currentDateTime())

        layout.addWidget(self.datetime)

        # 使用两个button(ok和cancel)分别连接accept()和reject()槽函数

        buttons = QDialogButtonBox(

            QDialogButtonBox.Ok | QDialogButtonBox.Cancel,

            Qt.Horizontal, self)

        buttons.accepted.connect(self.accept)

        buttons.rejected.connect(self.reject)

        layout.addWidget(buttons)

    # 从对话框中获取当前日期和时间

    def dateTime(self):

        return self.datetime.dateTime()

    # 静态方法创建对话框并返回 (date, time, accepted)

    @staticmethod

    def getDateTime(parent=None):

        dialog = DateDialog(parent)

        result = dialog.exec_()

        date = dialog.dateTime()

        return (date.date(), date.time(), result == QDialog.Accepted)

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

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

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

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

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

import sys

from PyQt5.QtCore import *

from PyQt5.QtGui import *

from PyQt5.QtWidgets import *

from DateDialog2 import DateDialog

class WinForm(QWidget):

    def __init__(self, parent=None):

        super(WinForm, self).__init__(parent)

        self.resize(400, 90)

        self.setWindowTitle('信号与槽传递参数的示例')

        self.open_btn = QPushButton('获取时间')

        self.lineEdit_inner = QLineEdit(self)

        self.lineEdit_emit = QLineEdit(self)

        self.open_btn.clicked.connect(self.openDialog)

        self.lineEdit_inner.setText('接收子窗口内置信号的时间')

        self.lineEdit_emit.setText('接收子窗口自定义信号的时间')

        grid = QGridLayout()

        grid.addWidget(self.lineEdit_inner)

        grid.addWidget(self.lineEdit_emit)

        grid.addWidget(self.open_btn)

        self.setLayout(grid)

    def openDialog(self):

        dialog = DateDialog(self)

        '''连接子窗口的内置信号与主窗口的槽函数'''

        dialog.datetime_inner.dateTimeChanged.connect(self.deal_inner_slot)

        '''连接子窗口的自定义信号与主窗口的槽函数'''

        dialog.Signal_OneParameter.connect(self.deal_emit_slot)

        dialog.show()

    def deal_inner_slot(self, date):

        self.lineEdit_inner.setText(date.toString())

    def deal_emit_slot(self, dateStr):

        self.lineEdit_emit.setText(dateStr)

if __name__ == "__main__":

    app = QApplication(sys.argv)

    form = WinForm()

    form.show()

    sys.exit(app.exec_())

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

from PyQt5.QtCore import *

from PyQt5.QtGui import *

from PyQt5.QtWidgets import *

class DateDialog(QDialog):

    Signal_OneParameter = pyqtSignal(str)

    def __init__(self, parent=None):

        super(DateDialog, self).__init__(parent)

        self.setWindowTitle('子窗口:用来发射信号')

        # 在布局中添加部件

        layout = QVBoxLayout(self)

        self.label = QLabel(self)

        self.label.setText('前者发射内置信号\n后者发射自定义信号')

        self.datetime_inner = QDateTimeEdit(self)

        self.datetime_inner.setCalendarPopup(True)

        self.datetime_inner.setDateTime(QDateTime.currentDateTime())

        self.datetime_emit = QDateTimeEdit(self)

        self.datetime_emit.setCalendarPopup(True)

        self.datetime_emit.setDateTime(QDateTime.currentDateTime())

        layout.addWidget(self.label)

        layout.addWidget(self.datetime_inner)

        layout.addWidget(self.datetime_emit)

        # 使用两个button(ok和cancel)分别连接accept()和reject()槽函数

        buttons = QDialogButtonBox(

            QDialogButtonBox.Ok | QDialogButtonBox.Cancel,

            Qt.Horizontal, self)

        buttons.accepted.connect(self.accept)

        buttons.rejected.connect(self.reject)

        layout.addWidget(buttons)

        self.datetime_emit.dateTimeChanged.connect(self.emit_signal)

    def emit_signal(self):

        date_str = self.datetime_emit.dateTime().toString()

        self.Signal_OneParameter.emit(date_str)

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,634评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,951评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,427评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,770评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,835评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,799评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,768评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,544评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,979评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,271评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,427评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,121评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,756评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,375评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,579评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,410评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,315评论 2 352

推荐阅读更多精彩内容