# import PyQt

:Python 的第三方库,用于调用 Qt 的 API ,从而制作 GUI 软件。

# Qt

  • Qt 是一个 GUI 工具包。

    • 很多工具都可以开发 GUI 软件,而 Qt 擅长开发跨平台的 GUI 软件。
      • 使用 Qt 开发一个 GUI 软件之后,在不修改源代码的情况下,可以为 Windows、Linux、Android、MacOS、iOS 等平台,分别构建一个可执行文件。
      • 例如 Linux 常见的图形桌面 KDE 就是基于 Qt 开发的。
    • Qt 本身是用 C++ 开发的,因此最初只能用 C++ 开发 GUI 软件。但后来也可以用 C#、Python 等编程语言,调用 Qt 的 API ,从而开发 GUI 软件。
  • Qt 的版本 (opens new window)

    • 首次发布于 1995 年。
    • Qt 4.0
      • 于 2005 年发布。
      • Qt 4.7 添加了 Qt Quick 和 Qt QML 。
    • Qt 5.0
      • 于 2012 年发布。
      • 进行了重构,拆分为几十个模块。因此不兼容 Qt4 。
    • Qt 6.0
      • 于 2020 年发布。
      • 兼容 Qt5 。
  • Qt5 的常用模块:

    • Qt Core
      • :提供 signal、slot 等核心的 Qt 功能。
    • Qt GUI
      • :提供 QPainter、QColor 等基础的 GUI 功能。
    • Qt Widgets
      • :提供 Window、Frame、Label、Button 等多种类型的 widget 。
    • Qt Creator
      • :一个 IDE 。
    • Qt Designer
      • :一个可视化编辑器。用户可以用鼠标拖动 widget ,来编辑 GUI 界面。
      • 编辑结果,通常保存为一个 .ui 文件,用于描述 GUI 界面,采用 XML 语法。比如显示哪些 widget 、每个 widget 的样式。
    • Qt QML
      • Qt 最初采用 XML 语法来描述 GUI 界面,但它不方便人类阅读。于是 Qt 发明了一种新的语言,称为 QML ,比 XML 更简洁、功能更多。
    • Qt Quick
      • :提供一些动态的、可触摸的 widget ,更适合手机 app 。
      • 传统 Qt 开发模式,是用 Qt Designer 设计 GUI 界面,基于 Qt Widgets 显示各种 widget 。
      • 新的 Qt 开发模式,是用 QML 设计 GUI 界面,基于 Qt Quick 显示各种 widget 。本文不介绍 Qt Quick 。
    • qmake
      • :可以为 Windows、Linux 等不同平台,分别生成一个 Makefile 。
    • Qt Network
      • :用于进行网络通信,支持 TCP、UDP、HTTP 等协议。
    • Qt Multimedia
      • :用于播放多媒体,比如图片、音频、视频。
  • Qt 可以显示 Window、Frame、Label、Button 等多种类型的 GUI 元素,它们统称为控件(widget)。

    • 大部分 widget 之间存在父子关系。例如一个 Frame 可能包含多个 Label ,担任它们的父容器(parent)。
    • 当父控件被销毁时,它的所有子控件都会被自动销毁。
    • 一个 GUI 软件可能显示多个窗口。通常将第一个显示的窗口,称为主窗口。主窗口通常包含很多子控件。
    • 除了 Qt 自带的各种 widget ,用户也可以自定义 widget ,做成插件,然后便可以显示在 Qt Designer 左侧的 widget 列表中,也可以通过 "promoted widget" 功能导入。

# 启动

  • 执行以下代码,即可显示一个 GUI 窗口:
    from PyQt6.QtWidgets import QApplication, QWidget
    import sys
    
    app = QApplication(sys.argv)  # 创建一个 QApplication 对象,代表一个 GUI 程序
    window = QWidget()            # 创建一个窗口,作为主窗口
    window.show()                 # 显示窗口
    
    sys.exit(app.exec())          # 进入 app 的主循环,阻塞当前线程。一旦 app 结束,就终止当前 Python 进程
    
    • QApplication 采用单例模式。可以重复调用 QApplication.instance() ,获取对 app 的引用。
    • 同一 GUI 程序中,只能存在一个主窗口。如果重复调用 QWidget() 创建一个主窗口,则会自动销毁之前的主窗口。

# QtCore

# QEvent

  • 用户对 widget 进行的鼠标点击、键盘输入等操作,统称为事件(event)。

    • Qt 自带了多种 widget ,每种 widget 支持多种 event 。
  • 可以重载 widget 对某种 event 的处理方法。使得每次发生该 event 时,就执行特定的操作。

    • 例:
      class MyWindow(QWidget):
          def __init__(self):
              super().__init__()
          def keyPressEvent(self, event):
              print(event.key())    # 获取用户按下的按键键值,比如按键 A 的键值是 65(不受大小写的影响)
          def mouseMoveEvent(self, event):
              print(event.pos())    # 获取鼠标的相对坐标(以窗口左上角为原点,当鼠标移到窗口外部时,坐标可能为负数)
      
      • 默认情况下,当用户按住鼠标并移动时,才会触发 mouseMoveEvent 事件。如果设置 self.setMouseTracking(True) ,则只要用户在窗口范围内移动鼠标,就会触发 mouseMoveEvent 事件。

# Signal

  • widget 可以在发生某种 event 时,自动发送某种信号(signal),通知其它对象。
    • 用户可以将 widget 的某种信号,绑定到一个函数,称为槽(slot)函数。使得每次发生该信号时,就调用一次槽函数,从而执行特定的操作。
    • 每种 widget 自带了几种信号、槽函数。用户也可以增加新的信号、槽函数。
  • 例:
    • QPushButton 按钮提供了几种信号:
      >>> button = QPushButton('Close', window)
      >>> button.show()
      >>> button.pressed    # 用户在按钮区域,按下鼠标左键时,会触发该信号
      <bound PYQT_SIGNAL pressed of QPushButton object at 0x000001DBBDEBCC10>
      >>> button.clicked    # 用户在按钮区域,按下鼠标左键并释放时,会触发该信号
      <bound PYQT_SIGNAL clicked of QPushButton object at 0x000001DBBDEBCC10>
      
    • 调用信号的 connect() 方法,可以将该信号,绑定到一个函数。
      >>> button.clicked.connect(print)           # 绑定一个槽函数
      <PyQt6.QtCore.QMetaObject.Connection object at 0x000001DBBDEF4C10>
      >>> button.clicked.connect(window.close)    # 每种信号,可以绑定多个槽函数
      <PyQt6.QtCore.QMetaObject.Connection object at 0x000001DBBDEF4C80>
      
    • 可以将一个信号,绑定到另一个信号。使得前一个信号发生时,自动触发后一个信号。
      button2 = QPushButton('button2', window)
      button2.clicked.connect(button2.clicked)
      
    • 调用信号的 disconnect() 方法,可以解绑一个槽函数。(如果不存在该绑定关系,则会抛出异常)
      >>> button.clicked.disconnect(window.close)
      
    • 调用信号的 emit() 方法,可以主动触发一次该信号。
      >>> button.clicked.emit()
      
  • 例:自定义信号
    >>> from PyQt6.QtCore import pyqtSignal
    >>> class MyWindow(QWidget):
    ...     _signal = pyqtSignal(str)   # 自定义一个信号,并声明其各个形参的数据类型
    ...     def __init__(self):
    ...         super().__init__()
    ...         self._signal.connect(self._slot)
    ...         self._signal.emit('testing')
    ...     def _slot(self, string):
    ...         print(string)
    ...
    >>> window = MyWindow()
    testing
    

# QTime

  • :用于获取时间。
  • 例:
    >>> from PyQt6.QtCore import QDateTime, QDate, QTime
    >>> QDateTime.currentDateTime()
    PyQt6.QtCore.QDateTime(2020, 1, 12, 10, 56, 40, 638)
    >>> QDate.currentDate()
    PyQt6.QtCore.QDate(2020, 1, 12)
    >>> QTime.currentTime()
    PyQt6.QtCore.QTime(10, 57, 14, 447)
    >>> _.second()
    14
    
  • 也可使用 Python 自带的 time、datetime 模块,获取时间。

# QTimer

  • :定时器。用于在一段时间之后,执行特定操作。
  • 定义:
    from PyQt6.QtCore import QTimer
    QTimer(parent: QObject = None)
    
  • 例:
    >>> timer = QTimer(window)                  # 创建一个定时器
    >>> timer.timeout.connect(lambda:print(1))  # 当定时器到达目标时刻时,会发出 timeout 信号。这里给 timeout 信号绑定一个槽函数
    <PyQt6.QtCore.QMetaObject.Connection object at 0x000001F8DBC5DBA0>
    >>> timer.setInterval(1000)                 # 设置定时器的间隔时间,即多久发出一次 timeout 信号,单位为毫秒
    >>> timer.start()                           # 启动定时器。定时器会一直运行,循环发出 timeout 信号
    1                                           # 可以用timer.start(1000),在启动时设置间隔时间
    1
    1
    >>> timer.stop()                            # 停止定时器。此后可以用 timer.start() 再次启动
    
    >>> timer.isActive()          # 定时器是否激活,正在运行
    False
    >>> timer.remainingTime()     # 距离 timeout 的剩余时间
    -1
    >>> timer.setSingleShot(True) # 采用 SingleShot 模式,让定时器发生一次 timeout 之后,自动 stop
    >>> timer.isSingleShot()      # 判断是否采用 SingleShot 模式
    True
    

# QtGui

# QPainter

  • 用于绘制 2D 图像。

# QColor

  • :用于选择颜色。
  • 定义:
    from PyQt6.QtGui import QColor
    QColor(int, int, int, alpha: int = 255)
        # 前三个参数,代表 RGB 三个通道的值,取值范围为 0~255
        # alpha 参数,表示不透明度。最大为 255 ,表示完全不透明
    
  • 例:
    >>> color = QColor(0, 0, 255) # 选择颜色
    >>> color.name()              # 返回颜色的十六进制值
    '#0000ff'
    >>> color.isValid()           # 判断是否为有效的颜色值
    True
    
  • 可以单独读取 red、green、blue、alpha 通道,或者调用 setxxx() 进行赋值。
    >>> color.alpha()
    255
    >>> color.setAlpha(0)
    

# QFont

  • :用于选择字体。
  • 例:
    >>> from PyQt6.QtGui import QFont
    >>> QFont('微软雅黑', 12)        # 输入字体名称、字号
    <PyQt6.QtGui.QFont object at 0x0000023BE20F9358>
    

# QIcon

  • :用于显示图标。
    from PyQt6.QtGui import QIcon
    icon = QIcon(r'./1.jpg')
    button = QPushButton('test', parent=window)
    button.setIcon(icon)
    

# QPixmap

  • :用于显示图片。
  • 例:
    from PyQt6.QtGui import QPixmap
    pixmap = QPixmap(r'./1.jpg')
    label = QLabel(window)
    label.setPixmap(pixmap)   # 用图片填充 label ,作为背景图
    window.resize(pixmap.width(), pixmap.height())
    

# QtWidgets

# QWidget

:用于创建普通窗口。

  • QWidget是所有窗口的基类,是 QMainWindow、QDialog、QFrame 的父类。

  • 定义:

    from PyQt6.QtWidgets import QWidget
    
    window = QWidget(parent: QWidget= None)
        # 功能:创建一个窗口
        # 如果不指定 parent 父控件,则它会成为主窗口
    
  • 一般控件的通用方法:

    window.show() -> None
        # 功能:显示该控件
        # 父控件在第一次显示时会初始化布局,将它的各个子控件也显示出来
        # 如果在父控件初始化之后,才加入子控件,则需要主动调用子控件的 show() 方法
    
    window.hide() -> Bool
      # 功能:隐藏该控件的显示。但并没有销毁,还可以再次 show()
    
    window.close() -> Bool
      # 功能:关闭该控件。
      # 如果成功关闭,或者该控件本来就没有显示,则返回 True
      # close() 与 hide() 都会让控件不再显示。但 close() 还可能销毁控件的对象、数据
    
    window.setEnabled(bool) -> None
        # 功能:设置控件是否可以被用户操作,即接收用户的鼠标点击、键盘输入
        # 这会影响到它的所有子控件
    
    window.setToolTip(str) -> None
        # 功能:设置提示语
        # 当鼠标悬停在一个控件上方时,会显示其提示语
        # 可以设置字体,例如: QToolTip.setFont(QFont('微软雅黑', 12))
        # 可以使用 HTML 语法,例如: window.setToolTip('This is a <font color="red">tip</font>.')
    
    window.setFocus() -> None
        # 功能:使控件得到屏幕焦点
    
    window.isActiveWindow() -> bool
        # 功能:判断控件是否获得了屏幕焦点
    
    window.isVisible() -> bool
        # 功能:判断控件是否正在显示
        # 如果控件刚刚创建,尚未调用 show() ,则没有显示
        # 如果控件调用了 close() ,则没有显示
    
  • QWidget 的特有方法:

    window.setWindowTitle(str) -> None
        # 功能:设置窗口的标题
    
    window.setWindowIcon(QIcon) -> None
        # 功能:设置窗口的图标
    
    window.isMaximized() -> bool
        # 功能:判断窗口当前的显示,是否最大化
        # 窗口最大化时,不一定是全屏。比如限制了窗口最大尺寸时,不能填满屏幕
    
    window.isMaximized() -> bool
        # 功能:判断窗口当前的显示,是否最小化
    

# QMainWindow

  • QMainWindow 是 QWidget 的子类,增加了菜单栏、工具栏、状态栏,因此更擅长担任主窗口。

  • 定义:

    from PyQt6.QtWidgets import QMainWindow
    window = QMainWindow(parent: QWidget= None)
    
  • 菜单栏,是在窗口的顶部,显示一行动作按钮。

    • 菜单栏(menu bar)可以包含一组菜单(menu),每个菜单可以包含一组动作按钮。
    • 创建菜单栏:
      menubar = window.menuBar()  # 第一次调用将创建菜单栏,重复调用将返回菜单栏这个单例对象的引用
      
    • 往菜单栏中,添加菜单:
      menubar.addMenu(str)        -> QMenu  # 输入名字,添加一个菜单
      menubar.addMenu(QIcon, str) -> QMenu  # 输入图标和名字,添加一个菜单
      menubar.addMenu(QMenu)      -> QMenu  # 输入一个 QMenu 对象,添加一个菜单
      
    • 可以在一个菜单中,嵌套另一个菜单,称为子菜单。例:
      file_menu = menubar.addMenu('File')
      sub_menu = file_menu.addMenu('Open Recent File...')
      
    • 每个菜单,可以包含一组动作按钮:
      QMenu.addAction(str)        -> QAction
      QMenu.addAction(QIcon, str) -> QAction
      QMenu.addAction(QAction)    -> QAction
      
  • 例:添加一个动作按钮,会被鼠标单击触发

    from PyQt6.QtGui import QAction
    
    exit_action = QAction('Exit', window)
    exit_action.setShortcut('Ctrl+Q')                 # 设置快捷键
    exit_action.setStatusTip('Exit the application.') # 设置显示在状态栏的提示
    exit_action.triggered.connect(app.quit)           # 绑定到一个槽函数。当用户点击该按钮时,就会调用槽函数
    
    menubar = window.menuBar()
    file_menu = menubar.addMenu('File') # 创建菜单
    file_menu.addAction(exit_action)    # 添加动作
    
  • 例:添加一个动作按钮,可以勾选

    def debug_mode(state):    # 触发该槽函数时,会传入一个 state 参数,表示当前按钮是否被勾选
        if state:
            window.statusBar().showMessage('Debug mode is enabled')
        else:
            window.statusBar().showMessage('Debug mode is disabled')
    
    debug_action = QAction('Debug Mode', window, checkable=True)
    debug_action.setChecked(False)  # 设置按钮的初始状态,是否被勾选
    debug_action.triggered.connect(debug_mode)
    
    menubar = window.menuBar()
    fileMenu = menubar.addMenu('File')
    fileMenu.addAction(debug_action)
    
  • 工具栏,是在菜单栏下方显示一行常用的动作按钮,方便用户点击。

    window.exit_tool = window.addToolBar('Exit')
    window.exit_tool.addAction(exit_action)
    
  • 状态栏,是在窗口的底部显示一行文字,供用户查看。

    • 例:
      statuebar = window.statusBar()
          # 功能:第一次调用将创建状态栏,重复调用将返回状态栏这个单例对象的引用
      
      statuebar.showMessage(str, msecs: int = 0)
          # 功能:在状态栏显示一行字符串
          # msecs 参数表示显示多少毫秒。默认为 0 ,表示永久显示
      
  • 上下文菜单,是单击鼠标右键会显示的一个菜单。

    • 不止是 QMainWindow , QWidget 窗口都支持显示上下文菜单。
    • 定义上下文菜单,需要重载 contextMenuEvent() 方法。如下:
      from PyQt6.QtWidgets import QApplication, QMainWindow, QMenu
      import sys
      
      app = QApplication(sys.argv)
      
      class MyWindow(QMainWindow):
          def contextMenuEvent(self, event):
              # 创建一个菜单,以 self 作为父容器
              contextMenu = QMenu(self)
      
              # 在菜单中,添加动作按钮
              open_action = contextMenu.addAction('open')
              exit_action = contextMenu.addAction('exit')
              exit_action.triggered.connect(app.exit)
      
              # 在鼠标的当前坐标处,显示菜单
              # 当用户点击任意动作按钮,就会关闭菜单的显示,并返回该按钮的引用
              clicked_action = contextMenu.exec(self.mapToGlobal(event.pos()))
      
              # 如果没有绑定槽函数,也可通过以下代码,检查用户点击了哪个按钮,然后执行相应的操作
              # if clicked_action == open_action:
              #     pass
              # elif clicked_action == exit_action:
              #     app.quit()
      
      window = MyWindow()
      window.show()
      sys.exit(app.exec())
      

# QLabel

  • :标签,即一个只读的字符串。
  • 定义:
    from PyQt6.QtWidgets import QLabel
    QLabel(text: str, parent: QWidget = None)
    
  • 例:
    label = QLabel('hello', window)   # 这里如果不指定 parent=window ,则该 label 不会显示在主窗口中
    window.show()
    

# QLineEdit

  • :单行输入框,用于输入一行字符串。
  • 定义:
    from PyQt6.QtWidgets import QLineEdit
    QLineEdit(parent: QWidget = None)
    QLineEdit(contents: str, parent: QWidget = None)  # 如果输入 contents 参数,则会作为输入框的初始内容
    
  • 例:
    >>> lineEdit = QLineEdit(window)
    >>> lineEdit.show()
    >>> lineEdit.setText('hello')   # 设置输入框的内容
    >>> lineEdit.text()             # 获取输入框的内容
    'hello'
    >>> lineEdit.setMaxLength(10)   # 限制输入内容的最大长度,超过则无法输入
    >>> lineEdit.setReadOnly(True)  # 使输入框变成只读。不能编辑,但可以选中、复制
    >>> lineEdit.setEchoMode(QLineEdit.EchoMode.Password)  # 使输入框的内容,显示成 *
    >>> lineEdit.clear()            # 清空输入框
    
  • 常用信号:
    lineEdit.textChanged.connect(...)       # 当输入框的内容改变时(不管是被用户改变,还是被程序改变)
    lineEdit.editingFinished.connect(...)   # 当用户输入结束时(比如按下回车、焦点从输入框移动到其它控件)
    lineEdit.selectionChanged.connect(...)  # 当用户的选中范围改变时(如果没有选中一段字符串,则不会触发)
    

# QTextEdit

  • :多行输入框,用于输入多行字符串。

  • 定义:

    from PyQt6.QtWidgets import QTextEdit
    QTextEdit(parent: QWidget = None)
    QTextEdit(text: str, parent: QWidget = None)
    
  • 例:

    >>> textEdit = QTextEdit(window)
    >>> textEdit.show()
    >>> textEdit.setPlainText('hello')  # 在纯文本模式下,设置输入框的内容
    >>> textEdit.toPlainText()          # 在纯文本模式下,获取输入框的内容
    'hello'
    >>> textEdit.clear()  # 清空输入框
    
    • 当输入的字符串行数过多时,会自动显示一个垂直滚动条。
  • QTextEdit 支持 HTML、Markdown 语法,从而显示富文本。

    >>> textEdit.setHtml('<h1>标题一</h1>') # 在 HTML 模式下,设置输入框的内容
    >>> textEdit.toHtml()                  # 在 HTML 模式下,获取输入框的内容
    '<!DOCTYPE HTML PUBLIC '-//W3C//DTD HTML 4.0//EN' ...'
    >>> textEdit.toPlainText()             # 在屏幕上,会显示这段文本,而不是 HTML 源代码
    '标题一'
    
    >>> textEdit.setMarkdown('# 标题一')  # 在 Markdown 模式下,设置输入框的内容
    >>> textEdit.toMarkdown()            # 在 Markdown 模式下,获取输入框的内容
    '# 标题一\n\n'
    >>> textEdit.toPlainText()
    '标题一'
    
  • 关于插入文本:

    textEdit.append('hello')                # 在末尾附加内容(这会新增一行)
    
    from PyQt6.QtGui import QTextCursor
    textEdit.moveCursor(QTextCursor.Start)  # 将光标移动到最前面
    textEdit.moveCursor(QTextCursor.End)    # 将光标移动到最后面
    textEdit.insertPlainText('hello')       # 在光标处插入纯文本
    textEdit.insertHtml('<h1>标题一</h1>')  # 在光标处插入 html
    

# 关于按钮

# QPushButton

  • :普通按钮。

  • 定义:

    from PyQt6.QtWidgets import QPushButton
    QPushButton(parent: QWidget = None)             # 只指定父控件,则显示一个空白按钮
    QPushButton(text: str, parent: QWidget = None)  # 输入一个要显示的字符串,再指定父控件
    QPushButton(icon: QIcon, text: str, parent: QWidget = None) # 输入图标、字符串、父控件
    
  • 例:

    >>> button = QPushButton('Quit', window)
    >>> button.text()                     # 返回按钮中的字符串
    'Quit'
    >>> button.clicked.connect(app.quit)  # 将按钮按下的信号,绑定到一个槽函数
    

# QCheckBox

  • :勾选按钮。有 "选中"、"未选中" 两种状态。

  • 定义:

    from PyQt6.QtWidgets import QCheckBox
    QCheckBox(parent: QWidget = None)
    QCheckBox(text: str, parent: QWidget = None)
    
  • 例:

    def changeState(self, state):
        if state == Qt.Checked:
            print('on')
        else:
            print('off')
    
    >>> checkBox = QCheckBox('debug', w)
    >>> checkBox.stateChanged.connect(changeState)  # 将状态改变的信号,绑定到一个槽函数
    >>> checkBox.isChecked()                        # 判断按钮是否被选中了
    True
    >>> checkBox.setChecked(True)                   # 设置状态
    >>> checkBox.toggle()                           # 切换一次状态
    
  • 例:让普通按钮,保持在 "按下" 或 "未按下" 状态,像一个勾选按钮

    def changeState(pressed):
        if pressed:
            print('on')
        else:
            print('off')
    
    button = QPushButton('debug', window)
    button.setCheckable(True)                   # 使普通按钮可以保持状态
    button.clicked[bool].connect(changeState)   # 绑定信号
    button.toggle()                             # 切换一次状态
    

# QRadioButton

  • :单选按钮。

# QComboBox

  • :下拉列表。
  • 定义:
    from PyQt6.QtWidgets import QComboBox
    QComboBox(parent: QWidget = None)
    
  • 例:
    >>> combo = QComboBox(window)
    >>> combo.show()
    >>> combo.addItem('one')  # 添加一个选项,并设置该选项显示的字符串
    >>> combo.addItem('two')
    >>> combo.currentText()   # 获取用户当前选中的选项
    'one'
    

# QScrollBar

  • :滚动条,用于拖动显示区域。

# QSlider

  • :滑块,用于调整某个数值。

  • 定义:

    from PyQt6.QtWidgets import QSlider
    QSlider(parent: QWidget = None)
    QSlider(orientation, parent: QWidget = None)
    
    • Qt.Orientation 表示显示方向,有两种取值:
      from PyQt6.QtCore import Qt
      Qt.Orientation.Horizontal # 垂直方向
      Qt.Orientation.Vertical   # 水平方向
      
  • 例:

    >>> slider = QSlider(window)
    >>> slider.show()
    >>> slider.setValue(100)  # 设置数值,取值范围为 0~99 ,超出范围则自动赋值为最小值、最大值
    >>> slider.value()        # 获取数值
    99
    
  • 常用信号:

    slider.valueChanged.connect(print)    # 当滑块的数值改变时,会触发该信号,给槽函数传入当前的数值
    slider.sliderPressed.connect(print)   # 当用户单击滑块时(只能使用鼠标左键,而鼠标右键无效)
    slider.sliderReleased.connect(print)  # 当用户松开鼠标左键时
    

# 关于对话框

  • 对话框是一种小型窗口,通常只会短暂显示。
  • QDialog 是 QtWidget 的子类,是各种对话框窗口的父类。
    • 创建一个对话框时,会立即显示它,不会等到调用 show() 方法。
    • 显示一个对话框时,会阻塞当前线程,直到对话框被关闭。

# QInputDialog

  • 输入对话框,用于输入一行字符串。
  • 例:
    >>> from PyQt6.QtWidgets import QInputDialog
    >>> text, ret = QInputDialog.getText(window, 'Input Dialog', 'Please input your name:')
    >>> text, ret
    ('hello', True)
    
    • 显示该对话框时,用户可以点击 OK 或 Cancel 按钮,或右上角的 Close 按钮。
      • 如果用户点击了 OK 按钮,则返回的 ret 为 True ,并且 text 存储了用户输入的文本。
      • 如果用户点击了 Cancel 或 Close 按钮,则返回的 ret 为 False ,并且 text 总是为空。

# QColorDialog

  • 颜色对话框,用于选择一个颜色。
  • 例:
    >>> from PyQt6.QtWidgets import QColorDialog
    >>> color = QColorDialog.getColor(parent=window, title='Select Color')
    >>> color
    <PyQt6.QtGui.QColor object at 0x000001486A345E40>
    >>> color.name()
    '#ffffff'
    

# QFileDialog

  • :文件对话框,用于选择本机的文件或目录。
  • 选择一个文件:
    >>> from PyQt6.QtWidgets import QFileDialog
    >>> filename, _filter = QFileDialog.getOpenFileName(window, 'Open file', '.', '*.py') # 输入参数:父控件、对话框标题、打开的目录、筛选的文件名
    >>> filename, _filter   # 返回的 filename 和 _filter ,都是 str 型
    ('D:/test/1.py', '*.py')
    
  • 选择多个文件:
    >>> filenames, _filter = QFileDialog.getOpenFileNames(window, 'Open files', '.', '*.py')
    >>> filenames, _filter  # 返回的 filenames 是一个字符串列表
    (['D:/test/1.py', 'D:/test/2.py'], '*.py')
    
  • 选择文件的保存路径:
    >>> QFileDialog.getSaveFileName(window, 'Save to', '.', '*.py')
    ('D:/test/3.py', '*.py')
    
  • 选择一个目录:
    >>> QFileDialog.getExistingDirectory(window, 'Open Directory', '.')
    'D:/test'
    

# QMessageBox

  • :消息框,用于通知用户某个消息。

  • 提问框,有 Yes、No 两个选项。

    >>> from PyQt6.QtWidgets import QMessageBox
    >>> reply = QMessageBox.question(window, '标题', '内容')
    >>> reply == QMessageBox.StandardButton.Yes
    True
    >>> reply == QMessageBox.StandardButton.No # 如果用户点击 No 或 Close 按钮,则返回值为 No
    False
    
  • 提示框,只有一个 OK 选项。

    >>> reply = QMessageBox.information(window, '标题', '内容')
    >>> reply == QMessageBox.StandardButton.Ok # 不管用户点击 Ok 还是 Close 按钮,返回值都为 Ok
    True
    
  • 警告框,只有一个 OK 选项。

    >>> reply = QMessageBox.warning(w, '标题', '内容')
    
  • 错误框:只有一个 OK 选项。

    >>> reply = QMessageBox.critical(w, '标题', '内容')
    

# 关于时间

  • 相关 widget :
    QDateEdit(parent: QWidget = None)
        # 功能:显示单行输入框,只能输入年月日格式的字符串
    
    QTimeEdit(parent: QWidget = None)
        # 功能:显示单行输入框,只能输入 24 小时格式的字符串
    
    QDateTimeEdit(parent: QWidget = None)
        # 功能:显示单行输入框,只能输入年月日 + 24 小时格式的字符串
    
    QCalendarWidget(parent: QWidget = None)
        # 功能:显示一个日历,用户通过鼠标单击,即可选择一个日期
    

# QProgressBar

  • :进度条。
  • 定义:
    from PyQt6.QtWidgets import QProgressBar
    QProgressBar(parent: QWidget = None)
    
  • 例:
    progressBar = QProgressBar(window)
    progressBar.setValue(80)  # 设置进度值。取值范围为 0~100 。如果输入为 float 类型,则小数部分会被舍去
    # progressBar.value()     # 返回当前的进度值
    window.show()
    

# 样式

# 坐标

  • 例:获取控件的坐标

    >>> pos = window.pos()  # 返回控件的坐标
    >>> pos                 # 取值为一个 QPoint 对象
    PyQt6.QtCore.QPoint(10, 10)
    >>> pos.x()   # 横坐标
    10
    >>> pos.y()   # 纵坐标
    10
    
    • 主窗口的坐标,是其左上角,相对于屏幕左上角,的相对坐标。
    • 普通控件的坐标,是其左上角,相对于父控件左上角,的相对坐标。
    • 主窗口可能移动到屏幕的显示区域之外,普通控件可能移动到父控件的显示区域之外,因此坐标可能为负数。
  • 例:转换坐标系:

    >>> window.mapToGlobal(QPoint(0, 0))    # 输入一个相对坐标(相当于当前控件),转换成全局坐标
    PyQt6.QtCore.QPoint(10, 41)             # 全局坐标的原点,并不是 (0,0) 开始,因为计算机屏幕存在边框
    >>> window.mapFromGlobal(QPoint(0, 0))  # 输入一个全局坐标,转换成相对坐标(相当于当前控件)
    PyQt6.QtCore.QPoint(-10, -41)
    
  • 例:移动控件到指定坐标

    >>> window.move(10, 10) # 输入横坐标 x、纵坐标 y ,取值为 int 类型,单位为 pixel
    
    >>> from PyQt6.QtCore import QPoint
    >>> window.move(QPoint(10, 10)) # 也可以输入一个 QPoint 对象
    

# 尺寸

  • 例:获取控件的尺寸

    >>> size = window.size()
    PyQt6.QtCore.QSize(200, 100)
    >>> size
    >>> size.width()
    200
    >>> size.height()
    100
    
    • 对于窗口类型的控件,height 不是指整个窗口的高度,而是指标题栏下方区域的高度。
  • 例:修改控件的尺寸

    >>> window.resize(400, 200) # 输入宽度 width 、高度 height ,取值为 int 类型,单位为 pixel
    
    >>> from PyQt6.QtCore import QSize
    >>> window.resize(QSize(400, 200)) # 也可以输入一个 QSize 对象
    
    • 对于窗口类型的控件,最小宽度 width 为 120 ,因为标题栏需要显示三个按钮。
    • 控件的尺寸,不能设置为 0 或负数,否则控件会关闭显示。
    • 控件的尺寸,不能超过屏幕的可用显示范围。即使用 resize() 设置了更大的尺寸也不会生效,即使用户用鼠标拖动窗口边框也不能变得更大。
  • 其它方法:

    # 根据控件需要显示的内容,自动调整尺寸
    window.adjustSize() -> None
    
    # 设置最大尺寸
    window.setMaximumHeight(int)    -> None
    window.setMaximumWidth(int)     -> None
    window.setMaximumSize(int, int) -> None
    window.setMaximumSize(QSize)    -> None
    
    # 设置最小尺寸
    window.setMinimumSize(int, int) -> None
    window.setMinimumSize(QSize)    -> None
    window.setMinimumHeight(int)    -> None
    window.setMinimumWidth(int)     -> None
    
    # 设置固定尺寸。此时用户将鼠标移动到窗口边框处,不能调整尺寸。只能由程序来调整尺寸
    window.setFixedSize(int, int)   -> None
    window.setFixedSize(QSize)      -> None
    window.setFixedHeight(int)      -> None
    window.setFixedWidth(int)       -> None
    
  • 调用控件的 geometry() 方法,可以同时获取坐标、尺寸。

    >>> geometry = window.geometry()
    >>> geometry
    PyQt6.QtCore.QRect(0, 0, 400, 200)
    >>> geometry.x()
    0
    >>> geometry.y()
    0
    >>> geometry.width()
    400
    >>> geometry.height()
    200
    
    window.setGeometry(0, 0, 400, 200)
    

# Layout

  • 布局(Layout),是指考虑一个窗口中,包含哪些 widget ,以及它们的坐标、尺寸。

  • Qt 提供了一些布局方法,用于自动调整 widget 的坐标、尺寸。

    QHBoxLayout   # 水平布局(Horizontal Layout),使多个控件在水平方向均匀排列
    QVBoxLayout   # 垂直布局(Vertical Layout),使多个控件在垂直方向均匀排列
    QGridLayout   # 网格布局,将显示空间分成多行多列,然后将控件放在某个格子内,或者占据多个格子
    QFormLayout   # 表格布局
    
    QFrame        # 框架,是一个透明的矩形框。通常给一个 QFrame 设置布局方法,然后容纳多个控件,方便统一管理
    
    • 自动布局通常比手动布局更好,因为可以在窗口放大、缩小时,按比例缩放各个 widget 。
    • 一个布局,可以容纳多个 widget 。
    • 一个布局,可以作为一个元素,嵌套放入另一个布局中。
  • 例:使用水平布局、垂直布局

    from PyQt6.QtWidgets import QApplication, QWidget, QPushButton, QHBoxLayout, QVBoxLayout
    import sys
    
    app = QApplication(sys.argv)
    window = QWidget()
    window.show()
    
    hbox = QHBoxLayout()    # 创建一个水平布局
    hbox.addStretch(1)      # 添加一个伸展因子。它像一个透明的弹簧,会自动挤占空白空间。取值越大,弹力越大
    hbox.addWidget(QPushButton('one')) # 添加一个控件。这些控件会按添加的先后顺序排列
    hbox.addWidget(QPushButton('two'))
    hbox.addStretch(1)      # 两侧各有一个 stretch ,就会将按钮挤到中间
    
    window.setLayout(hbox)  # 让 window 采用 hbox 布局,此时会显示该布局的所有控件,此后不能更换 window 的布局
    
    vbox = QVBoxLayout()    # 创建一个垂直布局
    vbox.addStretch(1)
    vbox.addWidget(QPushButton('three'))
    hbox.addLayout(vbox)    # 添加一个嵌套的布局
    
    • QMainWindow 预先划分了布局,在工具栏与状态栏之间,存在一个 central 区域。因此不能调用 setLayout() 方法,只能调用 setCentralWidget() 方法,将一个控件放置到 central 区域。
      window = QWidget()
      window.setLayout(hbox)
      
      main_window = QMainWindow()
      main_window.setCentralWidget(window)
      window.show()
      
    • 如果想从布局中删除一个控件,可以调用其 close() 方法。
  • 例:使用网格布局

    from PyQt6.QtWidgets import QApplication, QWidget, QPushButton, QGridLayout
    import sys
    
    app = QApplication(sys.argv)
    window = QWidget()
    window.show()
    
    grid = QGridLayout()
    window.setLayout(grid)
    # grid.setSpacing(10) # 设置每个格子的间距。而每个格子的尺寸,取决于此处的控件的尺寸
    
    grid.addWidget(QPushButton('one'), 0, 0)          # 添加一个控件,放在第 0 行、第 0 列
    grid.addWidget(QPushButton('one'), 0, 1)
    grid.addWidget(QPushButton('two'), 1, 1, 2, 2)    # 添加一个控件,占据从第 1 行、第 1 列,到第 2 行、第 2 列之间的格子
    
    • QGridLayout 会将显示空间分成多行多列。具体有多少行、多少列,取决于用户添加了多少个控件。
    • QGridLayout 会自动添加 stretch ,使得其中的控件,位置居中。而 QFormLayout 会使得其中的控件,位置趋于左上角。
  • 例:使用网格布局,显示一个计算器

    positions = [(i, j) for i in range(5) for j in range(4)]
    keys = [
        ['Clear', 'Back', '', 'Close'],
        ['7', '8', '9', '/'],
        ['4', '5', '6', '*'],
        ['1', '2', '3', '-'],
        ['0', '.', '=', '+'],
    ]
    for row_num,row in enumerate(keys):
        for column_num,key in enumerate(row):
            if key == '':
                continue
            button = QPushButton(key)
            grid.addWidget(button, row_num, column_num)
    

# StyleSheet

  • Qt 支持设置各个 widget 的样式(stylesheet),语法与 CSS 相似。
    • 官方文档 (opens new window)
    • 例:
      app.setStyleSheet("""
      QWidget{           # 作用于 QWidget 类的所有对象
          background-image: url(img/1.jpg);
      }
      QWidget:window {   # 作用于 QWidget 类,名为 window 的那个对象
          background-color: rgb(255, 255, 0);
          color: rgb(255, 85, 0);
      }
      """)
      
    • 子控件默认会继承父控件的 stylesheet 。

# 其它工具

# pyuic

  • Qt Designer 生成的 .ui 文件,如何在 Python 中使用?
    1. 先用 pyuic 工具,将 .ui 文件,转换成 .py 文件。如下:
      pyuic6 mainwindow.ui -o mainwindow_ui.py
      
    2. 然后在 mainwindow.py 中导入 mainwindow_ui.py :
      import sys
      from PyQt6.QtWidgets import QApplication, QMainWindow
      from mainwindow_ui import Ui_MainWindow
      
      class MyWindow(QMainWindow, Ui_MainWindow):  # 多继承
          def __init__(self):
              super().__init__()
              self.setupUi(self)
      
      app = QApplication(sys.argv)
      window = MyWindow()
      window.show()
      sys.exit(app.exec())
      

# pyrcc

  • Qt Designer 中如何导入图片?

    1. 创建一个 Label ,然后修改其 pixmap 属性,用一张图片填充该 Label 。
    2. 不能直接使用磁盘中的图片文件。需要先创建一个 .qrc 文件,用于记录当前 GUI 软件使用的各个资源文件的磁盘路径,采用 XML 语法。例如:
      <RCC>
        <qresource prefix="resource"> <!-- 一个前缀,可以被多个资源文件共享 -->
          <file>img/1.jpg</file>      <!-- 一个资源文件的磁盘路径(相对路径) -->
        </qresource>
      </RCC>
      
      然后在 .ui 文件中,引用 .qrc 文件中的图片:
      <pixmap resource="resource.qrc">:resource/img/1.jpg</pixmap>
      
  • .qrc 文件,如何在 Python 中使用?

    • 可用 pyrcc 工具,将 .qrc 文件转换成 .py 文件。这会将各个图片的二进制内容,存储在 .py 文件中。
      • 例如使用 PyQt5 时,执行:
        pyrcc5 resource.qrc -o resource.py
        
      • 使用 PyQt6 时,该工具改名为 pyside6-rcc ,并且需要安装 pip install PySide6
    • 然后可在 Python 代码中,引用 .qrc 文件中的图片:
      import resource
      QPixmap(r':resource/img/1.jpg')
      
    • 用 pyinstaller 打包时,需要指定各个 resource 文件的路径,从而加入打包:
      pyinstaller mainwindow.py -w -i resource/img/1.jpg