GUI编程之PyQt5入门详解(01)
GUI是Graphical User Interface的英文简称,即图形用户界面,准确地说,GUI就是屏幕产品的视觉体验和互动操作部分。GUI是一种结合计算机科学、美学、心理学、行为学及各商业领域需求分析的人机系统工程,强调人一机一环境三者作为一个系统进行总体设计。
⛄认识PyQt5
👀PyQt5框架简介
Python最初是作为一门脚本语言开发的,并不具备GUI功能,但由于其本身具有良好的可扩展性,能够不断地通过C/C++模块进行功能性扩展,因此目前已经有相当多的GUI控件集(Toolkit)可以在Python中使用了。Python中经常使用的GUI控件集有PyQt、Tkinter、wxPython、Kivy、PyGUI和Libavg,其中PyQt是Qt为Python专门提供的GUI扩展。
Qt是挪威Trolltech(奇趣科技公司)开发的一个C++ GUI应用程序,其包括跨平台类库、集成开发工具和跨平台IDE,既可以用于开发GUI程序,也可以用于开发非GUI程序。使用Qt只需开发一次应用程序,便可跨不同桌面和嵌入式操作系统部署该应用程序,而无须重新编写源代码。和Python一样,Qt也具有相当优秀的跨平台特性,使用Qt开发的应用程序能够在Windows、Linux和Mac OS平台之间轻松移植。
PyQt是一个用于创建GUI应用程序的跨平台的工具包,它将Python编程语言和Qt库成功融合在一起。Qt库是目前最强大的GUI库之一。PyQt是由Phil Thompson开发的,实现了一个Python模块集,拥有620多个类、将近6000个函数。PyQt可以运行在所有主流的操作系统上,包括UNIX、Windows和Mac OS。PyQt采用双许可证,开发人员可以选择GPL(GNU通用公共许可证)和商业许可。在此之前,GPL版本只能用在UNIX上;从PyQt4开始,GPL版本可用于所有支持的平台上。
博客:基础教程网站和书籍推荐
👀PyQt5的特点
PyQt5严格遵循Qt的发布许可,拥有双重协议,自由开发者可以选择使用免费的GPL版本,如果准备将PyQt用于商业活动,则必须为此交付商业许可费用。PyQt具有如下优秀的特性:
基于高性能的Qt的GUI控件集;
能够跨平台运行在Windows、Linux和Mac OS等系统上;
使用信号/槽(signal/slot)机制进行通信;
对Qt库的完全封装;
可以使用Qt成熟的IDE(如Qt Designer)进行图形界面设计,并自动生成可执行的Python代码;
提供了一整套种类繁多的窗口控件。
👀Qt与PyQt的关系
首先,PyQt是Qt框架的Python语言实现。PyQt提供了一个设计良好的窗口控件集合,每一个PyQt控件都有其对应的Qt控件。所以PyQt与Qt的类库和API非常详细,而且PyQt不再使用qmake系统和Q_OBJECT宏,使得PyQt再也没有编译链接错误,PyQt的代码也更加友好。
其次,在开发速度上,由于PyQt的核心就是Qt库,也是用C++编写的,所以即使逻辑代码运行速度慢一点,也不会成为性能瓶颈。在使用方式上,PyQt也没有失去Python的优雅语法、快速开发的能力。Python相对于C++的优点是在编程效率上,可以看到标准的Qt例子移植到PyQt后的代码具有相同的功能,使用相同的应用程序接口,Python版本的代码只有原来的50%~60%,而且更容易阅读。在开发效率上,由于Python是一种面向对象的语言,语法简单、高效,相对于C++而言,使用Python编写程序可以提高开发效率,减少开发成本。
最后,PyQt向Python程序员提供了使用完整的Qt应用程序接口的函数,几乎可以用Python做任何Qt能做的事。Qt和PyQt的设计都是完全面向对象的。Qt使用一种称为信号/槽的机制在窗口控件之间传递事件和消息。这种机制完全不同于其他图形界面开发库所采用的回调(callback)机制,使用信号/槽可以使程序更加安全和简洁。所开发的应用程序越大,Qt/PyQt的这个优势就越明显。
👀图形界面开发库介绍
从Python语言的诞生之日起,有许多优秀的GUI工具集被整合到Python当中,使得Python也可以在图形界面编程领域大展身手。
(1)Tkinter
Tkinter是绑定了Python的Tk GUI工具集,就是Python包装的Tcl代码,通过内嵌在Python解释器内部的Tcl解释器实现。将Tkinter的调用转换成Tcl命令,然后交给Tcl解释器进行解释,实现Python的GUI。Tk和其他语言的绑定,比如PerlTk,是直接由Tk中的C库实现的。
Tkinter是Python事实上的标准GUI,在Python中使用Tk GUI工具集的标准接口,已经包含在Python Windows安装程序中,著名的IDLE就是使用Tkinter实现GUI的。
(2)wxPython
wxPython是Python对跨平台的GUI工具集wxWidgets(用 C++编写)的包装,作为Python的一个扩展模块来实现。wxPython是比较流行的Tkinter的一个替代品,在各种平台上都表现良好。
(3)PyGTK
PyGTK是Python对GTK+GUI库的一系列包装。PyGTK是比较流行的Tkinter的一个替代品,Gnome下许多著名应用程序的GUI都是使用PyGTK实现的,比如BitTorrent、GIMP等。PyGTK和Gedit都有可选的实现,在Windows平台上似乎表现不太好,这一点也无可厚非,毕竟使用的是GTK的GUI库。
(4)PySide
PySide由Qt官方维护,是Python对跨平台的GUI工具集Qt的另一个包装,捆绑在Python当中。PySide是比较流行的Tkinter的一个替代品,拥有LGPL 2.1授权许可,允许进行免费的开源软件和私有的商业软件的开发。
在以上图形界面开发库中,前三个没有类似于Qt Designer(UI制作工具,它可以通过可视化操作创建UI文件,然后通过工具快速编译成Python文件,因此也可以把它视为一个代码生成器)的工具,所有的代码都需要手动输入,学习曲线非常陡峭;而第四个PySide本质上也是Qt的Python封装,只是支持Qt的版本比较老,而且官方已经停止维护这个库。所以,对于Python使用者来说,使用PyQt进行GUI开发是最好的选择。
⛄Qt Designer的使用
制作程序UI界面,一般可以通过UI制作工具和纯代码编写两种方式来实现。在PyQt5中,也可以采用这两种方式。本入门教程通过Qt Designer工具来制作UI界面。
👀Qt Designer快速入门
Qt Designer,即Qt设计师,是一个强大、灵活的可视化GUI设计工具,可以帮助我们加快开发PyQt程序的速度。它是专门用来制作PyQt程序中UI界面的工具,它生成的UI界面是一个后缀为.ui的文件,可以通过命令将.ui文件转换成.py格式的文件,并被其他Python文件引用。Qt Designer符合MVC(模型一视图一控制器)设计模式,做到了显示和业务逻辑的分离。Qt Designer具有以下优点。
使用简单,通过拖曳和点击就可以完成复杂的界面设计,而且还可以随时预览查看效果图。
转换Python文件方便。Qt Designer可以将设计好的用户界面保存为
.ui
文件,其实是XML 格式的文本文件。为了在PyQt中使用.ui
文件,可以通过 pyuic5命令将.ui
文件转换为.py
文件件,然后将.py
文件引入到自定义的Python代码中。
# 笔者机器上Qt Designer的安装路径 |
(1)新建主窗口
打开PyQt5的Qt Designer,会自动弹出“新建窗体”对话框。在模板选项中,最常用的就是Widget(通用窗口)和 Main Window(主窗口)。在PyQt5中Widget被分离出来,用来替代Dialog,并将Widget放入了QtWidget 模块库中。
(2)窗口主要区域
Widget Box(工具箱),提供了很多控件,包括常用的按钮、单选按钮、文本框等,可以直接拖放到主窗口中。在菜单栏中选择”窗体“→”预览“,或者按”Ctrl+R“快捷键,查看预览效果。
主窗口:即新建的窗口,界面设计的主窗口;
对象查看器:可以查看主窗口中放置的对象列表;
属性编辑器:提供了对窗口、控件、布局的属性编辑功能;
信号/槽编辑器、动作编辑器和资源浏览器:在信号/槽编辑器中,可以为控件添加自定义的信号和槽函数,编辑控件的信号和槽函数;在资源浏览器中,可以为控件添加图片,比如Label、Button的背景图片。
(3)查看UI文件
采用Qt Designer工具设计的界面文件默认为.ui
文件,描述了窗口中控件的属性列表和布局显示。.ui
文件里面包含的内容是按照XML(可扩展标记语言)格式处理的,文件中包含的参数信息与使用Qt Designer打开.ui
文件时显示的信息是一致的。
(4)将.ui
文件转换为.py
文件
使用Qt Designer设计的用户界面默认保存为.ui
文件,其内容结构类似于XML,但这种文件并不是我们想要的,我们想要的是.py
文件,所以还需要使用其他方法将.ui
文件转换为.py
文件。pyuic5官方网站
第一种方法:通过编译器将
.ui
文件转换为.py
;第二种方法:通过命令行将
.ui
文件转换为.py
。PyQt5安装成功后,pyuic5
命令行工具默认安装在Scripts
文件夹下。
# 笔者机器上pyuic5命令行工具的路径 |
- 第三种方法:通过Python脚本
.ui
文件转换为.py
。该脚本本质上是用Python代码把上述命令行操作封装起来,只要把该Python文件放在需要转换界面文件的目录下,直接运行即可,其执行效果和直接执行转换命令是一样的。其完整代码如下:
import os |
(5)界面与逻辑分离
由.ui
文件编译而来的.py
文件称为界面文件,由于界面文件每次编译时都会初始化,所以需要新建一个.py
文件调用界面文件,这个新建的.py
文件被称为逻辑文件,也可以称为业务文件。界面文件和逻辑文件是两个相对独立的文件,通过上述方法就实现了界面与逻辑的分离(也就是我们之前所说的“显示和业务逻辑的分离”)。实现界面与逻辑的分离方法很简单,只需要新建一个.py
文件,并继承界面文件的主窗口类即可。其完整代码如下:
import sys |
上面的代码实现了业务逻辑,代码结构也比较清晰。如果以后想要更新界面,只需要对.ui
文件进行更新,然后再编译成对应的.py
文件即可;而逻辑文件则视情况做一些调整,一般情况下不需要调整太多。
👀布局管理入门
(1)布局类型
Qt Designer提供了4种窗口布局方式,它们位于Qt Designer主窗口左侧区域的Widget Box(工具箱)里的Layouts(布局)栏中,分别是:
Vertical Layout(垂直布局):控件默认按照从上到下的顺序进行纵向添加。
Horizontal Layout(水平布局):控件默认按照从左到右的顺序进行横向添加。
Grid Layout(柵格布局):将窗口控件放入一个网格之中,然后将它们合理地划分成若干行(row)和列(column),并把其中的每个窗口控件放置在合适的单元(cell)中,这里的单元即是指由行和列交叉所划分出来的空间。
Form Layout(表单布局):控件以两列的形式布局在表单中,其中左列包含标签,右列包含输入控件。
一般进行布局有两种方式:一是通过布局管理器进行布局;二是通过容器控件进行布局。所谓容器控件,就是指能够容纳子控件的控件。使用容器控件,目的是将容器控件中的控件归为一类,以有别于其他控件。使用容器进行控件布局本质上还是调用布局管理器进行的。
(2)Qt Designer布局的顺序
使用Qt Designer开发一个完整的GUI程序流程如下:
将一个窗口控件拖入窗口中并放置在大致正确的位置上。除了容器(container)窗口,一般不需要调整窗口的尺寸大小。
对于要用代码引用的窗口控件,应指定一个名字:对于需要微调的窗口控件,可以设置其对应的属性。
重复步骤1和2,直到所需要的全部窗口控件都放到了窗口中。
如有需要,在窗口控件之间可以用Vertical Spacer、Horizontal Spacer、Horizontal Line、Vertical Line隔开(实际上前两步就可以包含这部分内容)。
选择需要布局的窗口控件,使用布局管理器或者切分窗口(splitter)对它们进行布局。
重复步骤5,直到所有的窗口控件和分隔符都布局好为止。
单击窗口,并使用布局管理器对其进行布局。
为窗口的标签设置伙伴关系。
如果按键次序有问题,则需要设置窗口的Tab键次序。
在适当的地方为内置的信号和槽建立信号与槽连接。
预览窗口(Ctrl+R),并检查所有的内容能否按照设想进行工作。
设置窗口的对象名(在类中会用到这个名字)、窗口的标题并进行保存。
编译窗口(命令行中使用pyuic5),根据需要生成对话框代码。
编写代码,即业务逻辑文件。
设置伙伴关系:即是控件之间的相互关联,使得可以通过一个控件控制另一个控件;
设置Tab键次序:程序中,在一个控件上使用Tab键会跳到另一个控件上,其跳转的规则或顺序即为Tab顺序。
伙伴关系设置是单方向的,即设置后只是单方向的控制。伙伴关系一般包含两个控件,暂且称为控制控件A和被控制控件B,即可以由控件A来控制控件B,一般情况下控件A(即控制控件)只能是Label控件。其他的不能设置伙伴关系,控件B(即被控制控件)没有要求。由于伙伴关系中的控制控件只能是Label控件,而Label控件在程序中无法选中或单击,故而通常给Label设置热键。
快捷键:在全局可以直接使用的键及组合,例如:选中控件使用(ctrl+1)即可完成水平布局,其为快捷键。
热键:相对于快捷键有所区别。热键是只有在打开了某个窗口时,出现了下拉菜单,只有在这个菜单展开的时候才可使用“快捷键”调用相应功能。
👀信号和槽关联
信号(signal)和槽(slot)是Qt的核心机制。在创建事件循环之后,通过建立信号和槽的连接就可以实现对象之间的通信。当信号发射(emit)时,连接的槽函数将会自动执行。
在PyQt5中,信号和槽通过QObject.signal.connect()连接。
注意:使用QObject.signal.connect()连接的槽函数不要加括号,否则会出错。
所有从QObject 类或其子类(如QWidget)派生的类都能够包含信号和槽。当对象改变其状态时,信号就由该对象发射出去。槽用于接收信号,但它们是普通的对象成员函数。多个信号可以与单个槽进行连接,单个信号也可以与多个槽进行连接。总之,信号和槽构建了一种强大的控件编程机制。
在Qt编程中,通过Qt信号槽机制对鼠标或键盘在界面上的操作进行响应处理,例如对鼠标单击按钮的处理。Qt中的控件能够发射什么信号,以及在什么情况下发射信号,在Qt的文档中有说明,不同的控件能够发射的信号种类和触发时机也是不同的。那么如何为控件发射的信号指定对应的处理槽函数呢?一般有三种方法:
第一种是在窗口的UI设计中操作添加信号和槽;
第二种是通过代码连接信号和槽;
第三种是通过集成开发环境IDE和编译器产生信号和槽。
👀菜单栏和工具栏
MainWindow即主窗口,主要包含菜单栏、工具栏、任务栏等。双击菜单栏上的”在这里输入“,然后输入文字,最后按回车键即可生成菜单。
子菜单可以通过动作编辑器或者属性编辑器中的Shortcut来添加快捷键。
动作编辑器中,双击需要编辑的动作,可以对其进行设置并添加图标、快捷键等。
默认生成的主窗口是不显示工具栏的,可以通过单击鼠标右键来添加工具栏;工具栏上的图标,可以通过动作编辑器建立并拖入工具栏中。
加载其他窗口,在当前窗口中嵌套另一个窗口。①首先创建第二个窗口,完成窗口设计,将
.ui
文件转换为.py
文件;②在主窗口的业务逻辑.py
中,创建类,集成子界面文件的主窗口类;③在主窗口的主界面类中生成子窗口实例,创建触发事件(信号→槽函数);④定义槽函数,添加子窗口。
👀资源文件打包
使用PyQt5生成的应用程序引用图片资源主要有两种方法,第一种方法是将资源文件转换为Python文件,然后引用Python文件;第二种方法是在程序中通过相对路径引用外部图片资源。这里以第一种方法为例:
(1)新建资源文件apprcc.qrc
<RCC> |
打开Qt Designer,进入资源浏览器界面,然后打开资源文件apprcc.qrc
,设置图片资源的前缀为pic,最后添加或删除图片资源。(它是以XML格式存储的)
<RCC> |
(2)在窗体中放置控件
在窗体中放置控件,并使用图片资源。将.ui
文件转化为.py
文件,运行.py
文件,会抛出如下异常信息,提示找不到apprcc_rc
。
ModuleNotFoundError: No module named 'apprcc_rc' |
(3)转换资源文件
使用pyrcc5
命令将apprcc.qrc
文件转换为apprcc_rc.py
文件(之所以添加_rc
,是因为Qt Designer导入资源文件时默认是加_rc
的)
pyrcc5 apprcc.qrc -o apprcc_rc.py |
转换完成后,在同级目录下会多出一个与*.qrc
文件同名的*_rc.py
(4)导入.py
资源文件
import apprcc_rc |
在Qt Designer中使用图片资源时,图片资源的引入路径是冒号:
加图片的路径,示例代码如下:
:/pic/image/002.jpg |
⛄PyQt5基本窗口控件
👀创建窗口类
QMainWindow、QWidget和QDialog三个类都是用来创建窗口的,可以直接使用,也可继承后再使用。
QMainWindow主窗口为用户提供一个应用程序框架,该窗口可以包含菜单栏、工具栏、状态栏、标题栏等,是最常见的窗口形式,是GUI程序的主窗口。
QWidget是所有用户界面对象的基类,所有的窗口和控件都直接或间接继承自QWidget类。QWidget是在PyQt中建立界面的主要元素,在PyQt中把没有嵌入到其他控件中的控件称为窗口,一般窗口都有边框、标题栏。窗口是指程序的整体界面,可以包含标题栏、菜单栏、工具栏、关闭按钮、最小化按钮、最大化按钮;控件是指按钮、复选框、文本框、表格、进度条等这些组成程序的基本元素。一个程序可以有多个窗口,一个窗口也可以有多个控件。
QDialog是对话框窗口的基类。对话框主要用来执行短期任务,或者与用户进行互动,它可以是模态的,也可以是非模态的。QDialog窗口没有菜单栏、工具栏、状态栏。
如果是主窗口,就使用QMainWindow类;如果是对话框,就使用QDialog类;如果不确定,或者有可能作为顶层窗口,也有可能嵌入到其他窗口中,那么就使用QWidget类。
👀QWidget
PyQt使用统一的坐标系统来定位窗口控件的位置和大小。以屏幕的左上角为原点,即(0, 0)点,从左向右为x轴正向,从上向下为y轴正向,整个屏幕的坐标系统就用来定位顶层窗口的。此外,在窗口内部也有自己的坐标系统,该坐标系统仍然以左上角作为原点,从左向右为x轴正向,从上向下为y轴正向,原点、x轴、y轴围成的区域叫作 Client Area(客户区),在客户区的周围则是标题栏(Window Title)和边框(Frame)。
- QWidget直接提供的成员函数:x()、y()获得窗口左上角的坐标,width()、height()获得客户区的宽度和高度。
- QWidget的geometry()提供的成员函数:x()、y()获得客户区左上角的坐标,width()、height()获得客户区的宽度和高度。
- QWidget的frameGeometry()提供的成员函数:x()、y()获得窗口左上角的坐标,width()、height()获得包含客户区、标题栏和边框在内的整个窗口的宽度和高度。
👀QLabel
QLabel对象作为一个占位符可以显示不可编辑的文本或图片,也可以放置一个GIF动画,还可以被用作提示标记为其他控件。纯文本、链接或富文本可以显示在标签上。其设置属性的方法和常用信号可以参考Qt Designer的属性编辑器和信号/槽编辑器。
👀文本框类控件
(1)QLineEdit类
QLineEdit类是一个单行文本框控件,可以输入单行字符串。如果需要输入多行宇符串,则使用QTextEdit类。其设置属性的方法和常用信号可以参考Qt Designer的属性编辑器和信号/槽编辑器,下面以几个为例:
setAlignment():按固定值方式对齐文本。
setPlaceholderText():设置文本框浮显文字。
setEchoMode():设置文本框显示格式(正常显示、密码类型输入、密码掩码字符等等)。
setValidator():设置文本框的验证器(验证规则),将限制任意可能输入的文本。比如整数、浮点数、正则表达式。
(2)QTextEdit类
QTextEdit类是一个多行文本框控件,可以显示多行文本内容,当文本内容超出显示范围时,可以显示水平垂直滚动条。QTextEdit不仅可以显示文本还可以显示HTML文档。其设置属性的方法和常用信号可以参考Qt Designer的属性编辑器和信号/槽编辑器。
setPlainText():设置多行文本框的文本内容;
toPlainText():返回多行文本框的文本内容;
setHtml():设置多行文本框的内容为HTML文档,HTML文档是描述网页的;
toHtml():返回多行文本框的HTML文档内容;
clear():清除多行文本框的内容。
👀按钮类控件
(1)QAbstractButton
在任何GUI设计中,按钮都是最重要的和常用的触发动作请求的方式,用来与用户进行交互操作。在PyQt中根据不同使用场景将按钮划分为不同表现形式。按钮的基类是QAbstractButton,提供了按钮的通用性功能。该基类为抽象类,不能实例化,必须由其它按钮类继承QAbstractButton类,来实现不同的功能、不同的表现形式。常见的按钮类包括:QPushButton、QToolButton、QRadioButton和QCheckBox。这些按钮类均继承自QAbstractButton类,根据各自的使用场景通过图形展现出来。QAbstractButton提供的状态如下:
状态 | 含义 |
---|---|
isDown() | 提示按钮是否被按下 |
isChecked() | 提示按钮是否已经标记 |
isEnable() | 提示按钮是否可以被用户点击 |
isCheckAble() | 提示按钮是否为可标记的 |
setAutoRepeat() | 设置按钮是否在用户长按时可以自动重复执行 |
QAbstractButton提供的信号如下:
信号 | 含义 |
---|---|
Pressed | 当鼠标指针在按钮上并按下左键时触发该信号 |
Released | 当鼠标左键被释放时触发该信号 |
Clicked | 当鼠标左键被按下然后释放时,或者快捷键被释放时触发该信号 |
Toggled | 当按钮的标记状态发生改变时触发该信号 |
(2)QPushButton
QPushButton类继承QAbstractButton类,形状是长方形,文本标题或图标可以显示在长方形上。它是一种命令按钮,可以单击该按钮执行一些命令,或响应一些事件。常见的有"确认"/"申请"/"取消"/"关闭"/"是"/"否"等按钮。命令按钮通常通过文本来描述执行的动作,有时候也会通过快捷键来执行对应按钮的命令。QPushButton类中的常用方法:
方法 | 描述 |
---|---|
setCheckable() | 设置按钮是否已经被选中,如果设置为True,则表示按钮将保持已点击和释放状态 |
toggle() | 在按钮状态之间进行切换 |
setIcon() | 设置按钮上的图标 |
setEnabled() | 设置按钮是否可以使用,当设置为False时,按钮变成不可用状态,点击不会发射信号 |
isChecked() | 返回按钮的状态。返回值为True或False |
setDefault() | 设置按钮的默认状态 |
setText() | 设置按钮的显示文本 |
text() | 返回按钮的显示文本 |
(3)QRadioButton
QRadioButton类继承QAbstractButton类,它提供一组可供选择的按钮和文本标签,用户可以选择其中一个选项,标签用于显示对应文本信息。单选钮是一种开关按钮,可以切换为on或者off,即checked或者unchecked,主要是为用户提供"多选一"的选择。
QRadioButton是单选钮控件默认是独占的(Exclusive)。对于继承自同一个父类Widget的多个单选钮,它们属于同一个按钮组合,在单选钮组里,一次只能选择一个单选钮。如果需要多个独占的按钮组合,则需要将它们放在 QGroupBox或QButtonGroup中。
当将单选钮切换到on或off时,会发送toggled信号,绑定这个信号,在按钮状态发生改变时,触发相应的行为。在QRadioButton中,toggled信号是在切换单选钮状态(开、关)时发射的,而clicked信号则在每次点击单选钮时都会发射。在实际中,一般只有状态改变时才有必要去响应,因此toggled信号更适合状态监控。QRadioButton类中的常用方法如下:
方法 | 描述 |
---|---|
setCheckable() | 设置按钮是否已经被选中,可以改变单选钮的选中状态,如果设置为True,则表示单选钮将保持已点击和释放状态 |
isChecked() | 返回单选钮的状态。返回值为True或False |
setText() | 设置单选钮的显示文本 |
text() | 返回单选钮的显示文本 |
(4)QCheckBox
QCheckBox类继承QAbstractButton类,提供一组带文本标签的复选框,用户可选择多个选项。和QPushButton一样,复选框可以显示文本或者图标,其中文本可以通过构造函数或者setText()来设置;图标可以通过setIcon()来设置。在视觉上,QButtonGroup可以把许多复选框组织在一起。
QCheckBox(复选框)和QRadioButton(单选钮)都是选项按钮,因为它们都可以在开(选中)或者关(未选中)之间切换。它们的区别是对用户选择的限制:单选钮提供了"多选一"的选择;而复选框提供的是"多选多"的选择。QCheckBox通常被应用在需要用户选择一个或多个可用的选项的场景中。
只要复选框被选中或取消选中,都会发射stateChanged信号。如果想在复选框状态改变时触发相应的行为,请连接这个信号,可以使用isChecked()来查询复选框是否被选中。除了常用的选中和未选中两种状态,QCheckBox还提供了第三种状态(半选中)来表明"没有变化"。当需要为用户提供一个选中或者未选中复选框的选择时,这种状态是很有用的。如果需要第三种状态,则可以通过setTristate()来使它生效,并使用checkState()来查询当前的切换状态。QCheckBox类中的常用方法如下:
方法 | 描述 |
---|---|
setChecked() | 设置复选框的状态,设置为True时表示选中复选框,设置为False时表示取消选中复选框 |
setText() | 设置复选框的显示文本 |
text() | 返回复选框的显示文本 |
isChecked() | 检查复选框是否被选中 |
setTristate() | 设置复选框为一个三态复选框 |
三态复选框有三种状态:
名称 | 值 | 含义 |
---|---|---|
Qt.Checked | 2 | 组件没有被选中(默认值) |
Qt.PartiallyChecked | 1 | 组件被半选中 |
Qt.Unchecked | 0 | 组件被选中 |
👀QComboBox(下拉列表框)
QComboBox是一个集按钮和下拉选项于一体的控件,也被称为下拉列表框。QComboBox类中的常用方法如下:
方法 | 描述 |
---|---|
addItem() | 添加一个下拉选项 |
addItems() | 从列表中添加下拉选项 |
Clear() | 删除下拉选项集合中的所有选项 |
count() | 返回下拉选项集合中的数目 |
currentText() | 返回选中选项的文本 |
itemText(i) | 获取索引为i的item的选项文本 |
currentIndex() | 返回选中项的索引 |
setItemText(int index, text) | 改变序号为index项的文本 |
QComboBox类中的常用信号如下:
信号 | 含义 |
---|---|
Activated | 当用户选中一个下拉选项时发射该信号 |
currentIndexChanged | 当下拉选项的索引发生改变时发射该信号 |
highlighted | 当选中一个已经选中的下拉选项时,发射该信号 |
👀QSpinBox(计数器)
QSpinBox是一个计数器控件,允许用户选择一个整数值,通过单击向上/向下按钮或按键盘上的上/下箭头来增加/减少当前显示的值,当然用户也可以输入值。
在默认情况下,QSpinBox的取值范围是0~99,每次改变的步长值为1。QSpinBox类和QDoubleSpinBox类均派生自QAbstractSpinBox类。QSpinBox用于处理整数值,QDoubleSpinBox用于处理浮点值,它们之间的区别就是处理数据的类型不同,其他功能都基本相同。QDoubleSpinBox的默认精度是两位小数,但可以通过setDecimals()来改变。QSpinBox类中的常用方法如下:
方法 | 描述 |
---|---|
setMinimum() | 设置计数器的下界 |
setMaximum() | 设置计数器的上界 |
setRange() | 设置计算器的最大值、最小值和步长值 |
setValue() | 设置计算器的当前值 |
Value() | 返回计数器的当前值 |
setSingleStep() | 设置计数器的步长值 |
每次单击向上/向下按钮时,QSpinBox计数器都会发射valueChanged信号,可以从相应的槽函数中通过value()函数获得计数器的当前值。
👀QSlider(滑动条)
QSlider控件提供了一个垂直或水平的滑动条,滑动条是一个用于控制有界值的典型控件,它允许用户沿水平或垂直方向在某一范围内移动滑块,并将滑块所在的位置转换成一个合法范围内的整数值。有时候这种方式比输入数字或者使用SpinBox(计数器)更加自然。在槽函数中对滑块所在位置的处理相当于从整数之间的最小值和最高值进行取值。一个滑块条控件可以以水平或垂直的方式显示,在构造函数中进行设置。
self.sp = QSlider(Qt.Horizontal) |
QSlider类中的常用方法如下:
方法 | 描述 |
---|---|
setMinimum() | 设置滑动条控件的最小值 |
setMaximum() | 设置滑动条控件的最大值 |
setSingleStep() | 设置滑动条控件递增/递减的步长值 |
setValue() | 设置滑动条控件的值 |
value() | 获取滑动条控件的值 |
setTickInterval() | 设置刻度间隔 |
setTickPosition() | 设置刻度标记的位置,可以输入一个枚举值,这个枚举值指定刻度线相对于滑块和用户操作的位置。以下是可以输入的枚举值: ①QSlider.NoTicks,不绘制任何刻度线; ②QSlider.TicksBothSides,在滑块的两侧绘制刻度线; ③QSlider.TicksAbove,在(水平)滑块上方绘制刻度线; ④QSlider.TicksBelow,在(水平)滑块下方绘制刻度线; ⑤QSlider.TicksLeft,在(垂直)滑块左侧绘制刻度线; ⑥QSlider.TicksRight,在(垂直)滑块右侧绘制刻度线。 |
QSlider类中的常用信号如下:
信号 | 描述 |
---|---|
valueChanged | 当滑块的值发生改变时发射此信号,此信号时最常用的 |
sliderPressed | 当用户按下滑块时发射此信号 |
sliderMoved | 当用户拖动滑块时发射此信号 |
sliderReleased | 当用户释放滑块时发射此信号 |
👀对话框类控件
(1)QDialog
为了更好地实现人机交互,比如Windows及Linux等系统均会提供一系列的标准对话框来完成特定场景下的功能,如选择字号大小、字体颜色等。在PyQt5中定义了一系列的标准对话框类,让使用者能够方便和快捷地通过各个类完成字号大小、字体颜色以及文件的选择等。
QDialog类的子类主要有QMessageBox、QInputDialog、QFontDialog、QFileDialog等。QDialog类中的常用方法如下:
方法 | 描述 |
---|---|
setWindowTitle() | 设置对话框标题 |
setWindowModality() | 设置窗口模态。取值如下: ①Qt.NonModal,非模态,可以和程序的其它窗口交互; ②Qt.WindowModal,窗口模态,程序在未处理完当前对话框时,将阻止和对话框的父窗口进行交互; ③Qt.ApplicationModal,应用程序模态,阻止和任何其他窗口进行交互。 |
当用户按下Esc键时,对话框窗口将会默认调用QDialog.reject()方法,然后关闭对话框窗口。
(2)QMessageBox
QMessageBox是一种通用的弹出式对话框,用于显示消息,允许用户通过单击不同的标准按钮对消息进行反馈。每个标准按钮都有一个预定义的文本、角色和十六进制数。QMessageBox类提供了许多常用的弹出式对话框,如提示、警告、错误、询问、关于等对话框。这些不同类型的QMessageBox对话框只是显示时的图标不同,其他功能是一样的。QMessageBox类中的常用方法如下:
方法 | 描述 |
---|---|
information(QWidget parent, title, text, buttons, defaultButton) | 弹出消息对话框,用来告诉用户关于提示信息,各参数解释如下: ①parent,指定的父窗口控件; ②title,对话框标题; ③text,对话框文本; ④buttons,多个标准按钮,默认OK按钮; ⑤defaultButton,默认选中的标准按钮,默认是第一个标准按钮。 |
question(QWidget parent, title, text, buttons, defaultButton) | 弹出问答/提问对话框(各参数解释同上),用来告诉用户关于提问信息。 |
warning(QWidget parent, title, text, buttons, defaultButton) | 弹出警告对话框(各参数解释同上),用来告诉用户关于不寻常的错误消息。 |
critical(QWidget parent, title, text, buttons, defaultButton) | 弹出严重错误对话框(各参数解释同上),用来告诉用户关于严重的错误信息。 |
about(QWidget parent, title, text) | 弹出关于对话框(各参数解释同上) |
setTitle() | 设置标题 |
setText() | 设置消息正文 |
setIcon() | 设置弹出对话框的图片 |
QMessageBox的标准按钮类型如下:
类型 | 描述 |
---|---|
QMessage.Ok | 同意操作 |
QMessage.Cancel | 取消操作 |
QMessage.Yes | 同意操作 |
QMessage.No | 取消操作 |
QMessage.Abort | 终止操作 |
QMessage.Retry | 重试操作 |
QMessage.Ignore | 忽略操作 |
(3)QInputDialog
QInputDialog控件是一个标准对话框,由一个文本框和两个按钮(OK按钮和Cancel按钮)组成。当用户单击OK或Enter键后,在父窗口可以收集通过QInputDialog控件输入的信息。QInputDialog控件是QDialog标准对话框的一部分。在QInputDialog控件中可以输入数字、字符串或列表中的选项,用于提示必要的信息。QInputDialog类中的常用方法如下:
方法 | 描述 |
---|---|
getInt() | 从控件中获得标准整数输入 |
getDouble() | 从控件中获得标准浮点数输入 |
getText() | 从控件中获得标准字符串输入 |
getItem() | 从控件中获得列表里的选项输入 |
(4)QFontDialog
QFontDialog控件是一个常用的字体选择对话框,可以让用户选择所显示文本的字号大小、样式和格式。此控件是QDialog标准对话框的一部分。使用QFontDialog类的静态方法getFont(),可以从字体选择对话框中选择文本的显示字号大小、样式和格式。
(5)QFileDialog
QFileDialog是用于打开和保存文件的标准对话框。QFileDialog类继承自QDialog类。QFileDialog在打开文件时使用了文件过滤器,用于显示指定扩展名的文件。也可以设置使用QFileDialog打开文件时的起始目录和指定扩展名的文件。QFileDialog类中的常用方法如下:
方法 | 描述 |
---|---|
getOpenFileName() | 返回用户所选择文件的名称,并打开该文件 |
getSaveFileName() | 使用用户选择的文件名并保存文件 |
setFileMode() | 可以选择的文件类型,枚举常量是: ①QFileDialog.AnyFile,任何文件; ②QFileDialog.ExistingFile,已存在的文件 ③QFileDialog.Directory,文件目录; ④QFileDialog.ExistingFiles,已经存在的多个文件 |
setFilter() | 设置过滤器,只显示过滤器允许的文件类型 |
👀窗口绘图类控件
在PyQt5中,一般可以通过QPainter、QPen和QBrush这三个类来实现绘图功能。此外,QPixmap的作用是加载并呈现本地图像,而图像的呈现本质上也是通过绘图方式实现的,所以QPixmap也可以被视为绘图的一个类。
(1)QPainter
QPainter类在QWidget(控件)上执行绘图操作,它是一个绘制工具,为大部分图形界面提供了高度优化的函数,使QPainter类可以绘制从简单的直线到复杂的饼图等。绘制操作在QWidget.paintEvent()中完成。绘制方法必须放在QtGui.QPainter对象的begin()和end()之间。QPainter类在控件或其他绘图设备上执行较低级别的图形绘制功能,并通过如下方法进行绘制。
方法 | |
---|---|
begin() | 开始在目标设备上绘制 |
drawArc() | 在起始角度和最终角度之间画弧 |
drawEllipse() | 在一个矩形内画一个椭圆 |
drawLine(int x1, int y1, int x2, int y2) | 绘制一条指定了端点坐标的线,绘制从(x1,y1)到(x2,y2)的直线并且 |
drawPixmap() | 从图像文件中提取Pixmap并将其显示在指定的位置 |
drawPolygon() | 使用坐标数组绘制多边形 |
drawRect(int x, int y, int w, int h) | 以给定的宽度w和高度h从左上角坐标(x,y)绘制一个矩形 |
drawText() | 显示给定坐标处的文字 |
fillRect() | 使用QColor参数填充矩形 |
setBrush() | 设置画笔风格 |
setPen() | 设置用于绘制的笔的颜色、大小和样式 |
还可以设置画笔风格(PenStyle),这是一个枚举类,可以由QPainter类绘制。画笔风格如表下所示:
枚举类型 | 描述 |
---|---|
Qt.NoPen | 没有线。比如QPainter.drawRect()填充,但没有绘制任何边界线 |
Qt.SolidLine | 一条简单的线 |
Qt.DashLine | 由一些像素分隔的短线 |
Qt.DotLine | 由一些像素分隔的点 |
Qt.DashDotLine | 轮流交替的点和短线 |
Qt.DashDotDotLine | 一条短线、两个点 |
Qt.MPenStyle | 画笔风格的掩码 |
(2)QPen
QPen(钢笔)是一个基本的图形对象,用于绘制直线、曲线或者给轮廓画出矩形、椭圆形、多边形及其他形状等。
(3)QBrush
QBrush(画刷)是一个基本的图形对象,用于填充如矩形、椭圆形或多边形等形状。QBrush有三种类型:预定义、过渡和纹理图案。
(4)QPixmap
QPixmap类用于绘图设备的图像显示,它可以作为一个QPaintDevice对象,也可以加载到一个控件中,通常是标签或按钮,用于在标签或按钮上显示图像。QPixmap可以读取的图像文件类型有BMP、GIF、JPG、JPEG、PNG、PBM、PGM、PPM、XBM、XPM等。QPixmap类中的常用方法如下:
方法 | 描述 |
---|---|
copy() | 从QRect对象复制到QPixmap对象 |
fromImage() | 将QImage对象转换为QPixmap对象 |
grabWidget() | 从给定的窗口小控件创建一个像素图 |
grabWindow() | 从窗口中创建数据的像素图 |
load() | 加载图像文件作为QPixmap对象 |
save() | 将QPixmap对象保存为文件 |
toImage() | 将QPixmap对象转换为QImage对象 |
👀拖曳与剪贴板
(1)Drag与Drop
为用户提供的拖曳功能很直观,在很多桌面应用程序中,复制或移动对象都可以通过拖曳来完成。基于MIME类型的拖曳数据传输是基于QDrag类的。QMimeData对象将关联的数据与其对应的MIME类型相关联。MIME类型的数据可以简单理解为互联网上的各种资源,比如文本、音频和视频资源等,互联网上的每一种资源都属于一种MIME类型的数据。MimeData类函数允许检测和使用方便的MIME类型如下:
判断函数 | 设置函数 | 获取函数 | MIME类型 |
---|---|---|---|
hasText() | text() | setText() | text/plain |
hasHtml() | html() | setHtml() | text/html |
hasUrls() | urls() | setUrls() | text/uri-list |
hasImage() | imageData() | setImageData() | image/* |
hasColor() | colorData() | setColorData() | application/x-color |
许多QWidget对象都支持拖曳动作,允许拖曳数据的控件必须设置QWidget.setDragEnabled()为True。另外,控件应该响应拖曳事件,以便存储所拖曳的数据。常用的拖曳事件如下:
事件 | 描述 |
---|---|
DragEnterEvent | 当执行一个拖曳控件操作,并且鼠标指针进入该控件时,这个事件将被触发。在这个事件中可以获得被操作的窗口控件,还可以有条件地接受或拒绝该拖曳操作 |
DragMoveEvent | 在拖曳操作进行时会触发该事件 |
DragLeaveEvent | 当执行一个拖曳控件操作,并且鼠标指针离开该控件时,这个事件将被触发 |
DropEvent | 当拖曳操作在目标控件上被释放时,这个事件将被触发 |
(2)QClipboard
QClipboard类提供了对系统剪贴板的访问,可以在应用程序之间复制和粘贴数据。它的操作类似于QDrag类,并使用类似的数据类型。QApplication类有一个静态方法clipboard(),它返回对剪贴板对象的引用。任何类型的MimeData都可以从剪贴板复制或粘贴。QClipboard类中的常用方法如下:
方法 | 描述 |
---|---|
clear() | 清除剪贴板的内容 |
setImage() | 将QImage对象复制到剪贴板中 |
setMimeData() | 将MIME数据设置为剪贴板 |
setPixmap() | 从剪贴板中复制Pixmap对象 |
setText() | 从剪贴板中复制文本 |
text() | 从剪贴板中检索文本 |
QClipboard类中的常用信号如下:
信号 | 含义 |
---|---|
dataChanged | 当剪贴板内容发生变化时,这个信号被发射 |
👀日历与时间
(1)QCalendar
QCalendar是一个日历控件,它提供了一个基于月份的视图,允许用户通过鼠标或键盘选择日期,默认选中的是今天的日期。也可以对日历的日期范围进行规定。QCalendar类中的常用方法如下:
方法 | 描述 |
---|---|
setDateRange() | 设置日期范围供选择 |
setFirstDayOfWeek() | 重新设置星期的第一天,默认是星期日。参数枚举(Qt.Monday ,Qt.Tuesday , Qt.Wednesday ,Qt.Thursday ,Qt.Friday ,Qt.Saturday ,Qt.Sunday ) |
setMaximumDate() | 设置最大日期 |
setMinimumDate() | 设置最小日期 |
setSelectedDate() | 设置一个QDate对象,作为日期控件所选定的日期 |
MaximumDate() | 获取日历控件的最大日期 |
MinimumDate() | 获取日历控件的最小日期 |
SelectedDate() | 返回当前选定的日期 |
setGridvisible() | 设置日历控件是否显示网格 |
(2)QDateTimeEdit
QDateTimeEdit是一个允许用户编辑日期时间的控件,可以使用键盘和上、下箭头按钮来增加或减少日期时间值。如当使用鼠标选中QDateTimeEdit中的年份时,可以使用键盘上的上、下键来改变数值。QDateTimeEdit通过setDisplayFormat()函数来设置显示的日期时间格式。QDateTimeEdit类中的常用方法如下:
方法 | 描述 |
---|---|
setDisplayFormat() | 设置日期时间格式。 ①yyyy,代表年份,用4位数表示 ②MM,代表月份,取值范围为01-12 ③dd,代表日,取值范围为01-31 ④HH,代表小时,取值范围为00-23 ⑤mm,代表分钟,取值范围为00-59 ⑥ss,代表秒,取值范围位00-59 |
setMaximumDate() | 设置控件的最大日期 |
setMinimumDate() | 设置控件的最小日期 |
time() | 返回编辑的时间 |
date() | 返回编辑的日期 |
QDateTimeEdit类中的常用信号如下:
信号 | 含义 |
---|---|
dateChanged | 当日期改变时发射此信号 |
dataTimeChanged | 当日期时间改变时发射此信号 |
timeChanged | 当时间改变时发射此信号 |
QDateEdit和QTimeEdit类均继承自QDateTimeEdit类,它们的许多特性和功能都由QDateTimeEdit类提供。
设置显示格式时要注意:QDateEdit用来编辑控件的日期,仅包括年、月和日;QTimeEdit用来编辑控件的时间,仅包括小时、分钟和秒。不要用QDateEdit来设置或获取时间,也不要用QTimeEdit来设置或获取日期。如果要同时操作日期时间,请使用QDateTimeEdit。
dateEdit = QDateEdit(self) |
设置弹出日历时要注意:用来弹出日历的类只有QDateTimeEdit和QDateEdit,而QTimeEdit类虽然在语法上可以设置弹出日历,但不起作用。使用它们弹出日历的正确方法如下。
dateTimeEdit = QDateTimeEdit(self) |
在默认情况下,如果QDateTimeEdit类构造时不指定日期时间,那么系统会为其设置一个和本地相同的日期时间格式,并且值为2000年1月1日0时0分0秒。也可以手动指定控件显示的日期时间。除通过构造函数指定所显示的日期时间外,也可以根据QDateTimeEdit提供的槽函数来设置,比如setDateTime()、setDate()、setTime()函数。
# (1)初始化QDateTimeEdit类 |
👀菜单栏、工具栏与状态栏
(1)菜单栏
在QMainWindow对象的标题栏下方,水平的QMenuBar被保留显示QMenu对象。QMenu类提供了一个可以添加到菜单栏的小控件,也用于创建上下文菜单和弹出菜单。每个QMenu对象都可以包含一个或多个QAction对象或级联的QMenu对象。要创建一个弹出菜单,PyQt API提供了createPopupMenu()函数;menuBar()函数用于返回主窗口的QMenuBar对象;addMenu()函数可以将菜单添加到菜单栏中;通过addAction()函数可以在菜单中进行添加操作。在设计菜单系统时使用的一些重要方法如下:
方法 | 描述 |
---|---|
menuBar() | 返回主窗口的QMenuBar对象 |
addMenu() | 在菜单栏中添加一个新的QMenu对象 |
addAction() | 向QMenu小控件中添加一个操作按钮,其中包含文本或图标 |
setEnabled() | 将操作按钮状态设置为启用/禁用 |
addSeparator() | 在菜单中添加一条分隔线 |
clear() | 删除菜单/菜单栏的内容 |
setShortcut() | 将快捷键关联到操作按钮 |
setText() | 设置菜单项的文本 |
setTitle() | 设置QMenu小控件的标题 |
text() | 返回与QAction对象关联的文本 |
title() | 返回QMenu小控件的标题 |
单击任何QAction按钮时,QMenu对象都会发射triggered信号。
# 菜单中的操作按钮可以是字符串或QAction对象 |
(2)QToolBar
QToolBar控件是由文本按钮、图标或其他小控件按钮组成的可移动面板,通常位于菜单栏下方。QToolBar类中的常用方法如下:
方法 | 描述 |
---|---|
addAction() | 添加具有文本或图标的工具按钮 |
addSeparator() | 分组显示工具按钮 |
addWidget() | 在工具栏中添加按钮以外的控件 |
addToolBar() | 使用QMainWindow类的方法添加一个新的工具栏 |
setMovable() | 工具栏变得可移动 |
setOrientation() | 工具栏的方向可以设置为Qt.Horizontal或Qt.vertical |
每当单击工具栏中的按钮时,都将发射actionTriggered信号。另外,这个信号将关联的QAction对象的引用发送到连接的槽函数上。
# 调用addToolBar()方法在工具栏区域添加文件工具栏 |
(3)QStatusBar
MainWindow对象在底部保留有一个水平条,作为状态栏(QStatusBar),用于显示永久的或临时的状态信息。 通过主窗口的QMainWindow的setStatusBar()函数设置状态栏,核心代码如下:
self.statusBar = QStatusBar() |
QStatusBar类中的常用方法如下:
方法 | 描述 |
---|---|
addWidget() | 在状态栏中添加给定的窗口小控件对象 |
addPermanentWidget() | 在状态栏中永久添加给定的窗口小控件对象 |
showMessage() | 在状态栏中显示一条临时信息指定时间间隔 |
clearMessage() | 删除正在显示的临时信息 |
removeWidget() | 从状态栏中删除指定的小控件 |
bar = self.menuBar() |
👀QPrinter
打印图像是图像处理软件中的一个常用功能。打印图像实际上是在QPaintDevice中画图,与平常在QWidget、QPixmap和QImage中画图一样,都是创建一个QPainter对象进行画图的,只是打印使用的是QPrinter,它本质上也是一个QPaintDevice(绘图设备)。