表格部件为应用程序提供标准的表格显示工具,在表格内可以管理基于行和列的数据项,表格中的最大数据项数为总行数和总列数的乘积,另外在表格中可以设置水平和垂直标题。
表格部件对应类为QTableWidget ,QTableWidget 表格部件中的项类型为QTableWidgetItem类。QTableWidget 从QTableView派生的子类,内置默认模型,如果表格展现的应用需要使用自己的数据模型,则应使用QTableView类,而不是QTableWidget 类。
除了从父类继承的属性外,在Designer中QTableWidget只有两个属性,就是行数rowCount和列数columnCount,另外还有一部分就是用于设置表头的属性。如图:
QTableWidget的rowCount属性保存表格部件中的行数,在QTableWidget创建时如果没有指定行数,则缺省行数为0,QTableWidget创建后可以通过 setRowCount方法调整行数。
要获取当前表格部件中的行数,可以通过rowCount()方法获取,要设置表格部件的行数,可以通过setRowCount(int rows)调整表格的行数,如果参数rows小于现在表格中的实际行数,则表格中超出参数的行数数据会丢弃,就算是后面将行数或列数恢复也不能恢复相关数据。
columnCount属性保存表格部件中的列数,相关属性和操作方法和rowCount类似。在QTableWidget创建时如果没有指定列数,则缺省列数为0,QTableWidget创建后可以通过 setColumnCount方法调整。
columnCount可以通过columnCount()方法获取,通过setColumnCount(int columns)设置。
QTableWidget的表头包括横表头和竖表头,如下:
上面的“列1”、“列2”、“列3”为横表头,“行1”、“行2”、“行3”为竖表头。横表头可通过方法setHorizontalHeaderLabels来设置,竖表头可以通过setVerticalHeaderLabels来设置。这是创建表头的最简单方法,它们将为表的列和行提供简单的文本标题。
下面代码为上图设置表头的示例代码:
self.tableWidget.setVerticalHeaderLabels(['行1','行2','行3'])
self.tableWidget.setHorizontalHeaderLabels(['列1','列2','列3'])
除了上述简单方法外,还可以从表格部件的项创建更复杂的表头,相关方法在下面介绍QTableWidget的方法时再介绍。
除了设置表头文字外,表头还有相关属性设置,QTableWidget的表头属性是直接继承自QTableView的表头属性,相关属性的介绍请参考《PyQt(Python+Qt)学习随笔:QTableView的标题表头相关属性》。
QTableWidgetItem类为QTableWidget类的项实例类,用于保存表格部件的信息。项的内容可包含文本、图标或复选框等。
默认情况下,QTableWidgetItem项是可用、可编辑、可选择和可选中的,并且可以用作拖放操作的源和拖放目标。如果要改变相关设置,可以通过使用setFlags()来更改每个项的标志(关于项的标记请参考《PyQt(Python+Qt)学习随笔:Model中项的标记flags取值及枚举类型Qt.ItemFlag》)。
QTableWidgetItem的方法分为构造方法、位置相关方法、数据及展现内容相关、操作相关四类。
QTableWidgetItem的构造方法有4个:
QTableWidgetItem项在QTableWidget中的位置包括三个属性来决定,就是表格部件对象、行和列,由于这三个属性决定了项在界面的哪个组件的哪个地方,因此老猿将相关方法归类为位置相关方法。QTableWidgetItem项方法中:
在部件中的数据有多种属性,包括文本数据、图标、复选状态、工具栏提示toolTip、whatsThis帮助、状态提示statusTip等,在Qt中将这些称为同一个项的不同角色数据(关于数据的角色请参考《PyQt学习随笔:Model/View中诸如DisplayRole的数据角色及含义》)。这些数据为呈现数据的不同状态情况,因此老猿将与此相关的方法归类为数据和展现相关方法。不同角色的数据在Qt中可以通过data和setData方法带角色参数统一访问,也可以通过各自不同的方法不带角色去访问。
部件中的项数据可以通过项的data( int role) 方法获取项中指定列指定角色的数据,也可以通过setData(int role, QVariant value)方法设置指定角色的数据为value。例如项的文本可以通过data方法和setData方法使用Qt.DisplayRole、Qt.EditRole这两种角色去访问。
关于数据的角色请参考《PyQt学习随笔:Model/View中诸如DisplayRole的数据角色及含义》。
item.setData(QtCore.Qt.DecorationRole,QtGui.QIcon('.\\icon\\'+str(row+1)+'.gif'))#设置图标
item.setData(QtCore.Qt.CheckStateRole, row%3) #设置复选状态
item.setData(QtCore.Qt.TextAlignmentRole,col % 3) #设置文本对齐方式
项的文本可以通过data方法和setData方法使用Qt.DisplayRole、Qt.EditRole这两种角色去访问,除了这个方法外,还可以通过text() 和setText( str value)方法去访问。
项中的文本可以设置水平和垂直对齐方式,相关对齐方式的访问方法调用方式如下:
对齐方式是一个Qt.Alignment枚举类型的组合,具体取值及含义请参考《PyQt(Python+Qt)学习随笔:QListView的itemAlignment属性》。
可以通过icon()来访问项的图标,通过setIcon( QIcon icon)来设置项的图标。
如下面代码将顶的图标设置为指定文件:
item.setIcon(QtGui.QIcon(r'F:\小图标\动物\动物-025.gif'))
项可以单独设置复选状态,如图所有项都设置了复选状态,每行的复选状态不同:
项的复选状态可以通过checkState()来获取,如果要改变项的复选状态可以调用setCheckState(Qt.CheckState state)来实施。
提示信息包括工具栏提示、状态栏提示和whatsThis提示:
关于这三个提示信息的区别请参考《PyQt(Python+Qt)学习随笔:Qt Designer中部件的toolTip、toolTipDuration、statusTip、whatsThis属性》。
关于QTableWidgetItem项的sizeHint属性,Qt中的文档说明非常简单,作用也没有展开说明,仅介绍QTableWidgetItem中sizeHint为项的缺省大小,如果没有设置则根据项的数据自动计算项的大小。
为此老猿做了大量测试和验证,最终确认了QTableWidgetItem项的sizeHint的作用,QTableWidgetItem的sizeHint在项对应QHeadView表头的sectionResizeMode值为ResizeToContents时,作为计算项大小的一个因素:
具体请参考《PyQt学习随笔:QTableWidget项sizeHint的作用以及与QHeadView的sectionResizeMode、ResizeToContents的关系》和《PyQt(Python+Qt)学习随笔:QTableWidget表格部件中行高和列宽的计算方式》。
QTableWidget中项操作相关的属性包括是否可用、是否可选中、是否可编辑、是否可复选、是否选中、是否复选等,这些属性的设置与影响界面上项的操作,所以老猿将其归类项操作相关方法。这些项操作相关方法所有Model/View相关类涉及项操作都是一样的。由于复选状态同时是项的展现状态,在前面数据和展现相关方法中已经介绍相关方法。
项的标记用于标记项是否可操作,由多个属性位组合而成,具体项标记的取值及含义请参考《PyQt(Python+Qt)学习随笔:Model中项的标记flags取值及枚举类型Qt.ItemFlag》。
项标记的访问方法及调用语法如下:
项是否被用户选中可以通过isSelected方法获取,如果要通过代码设置项的选中状态,则调用setSelected方法,相关调用语法如下:
除了从父类继承的方法外,QTableWidget的方法老猿将其归纳为构造方法、部件状态访问方法、项操作方法、项查找和定位方法、表头操作方法五大类。
QTableWidget有2个构造方法:
这两个构造方法的区别就是后者指定了部件的行数和列数,而前者行数和列数为0,需要在实例构建后再另外去指定行数和列数。参数parent 一般传部件所在窗口,不传也没关系。
这里的构造方法是沿用C++的说法,Python中的构造方法是__init__。
老猿将部件中反映部件当前情况的一些方法归类为部件状态访问方法,包括部件的行数、列数、当前项、当前行、当前列等属性访问方法。由于部件行数和列数的访问方法在部件属性中已经介绍了,在此不重复介绍。
当前项是指当前鼠标和键盘焦点所在项,在项可以进行选择操作时,当前项可以是选中状态,也可以是未选中状态,选中项也不一定是当前项。与当前项相关的方法包括:
当前项所在的行和列,既可以通过当前项的QTableWidgetItem实例对象的行号(row()方法)和列号(column()方法)获取,也可以直接通过QTableWidget获取。调用语法如下:
部件中的项操作包括设置项、删除项、编辑项等。
setItem用于在表格部件QTableWidget创建后,设定指定行和指定列的项为一个QTableWidgetItem实例对象。调用语法如下:
setItem(int row, int column, QTableWidgetItem item)
for row in range(5):
for col in range(3):
item = QtWidgets.QTableWidgetItem(f"({row},{col})" )
self.tableWidget.setItem(row,col,item)
QTableWidget提供了触发项编辑的方法,调用语法如下:
editItem(QTableWidgetItem item)
上面介绍editItem时说明了editItem只能触发一次编辑,可以说进入临时编辑状态,一旦退出编辑除非再通过相关方式触发编辑否则项不可再编辑。与此相对应,与QTreeWidget类似,QTableWidget还提供了一种一旦打开编辑状态就可以随时再次编辑,除非显示关闭编辑状态,这种方式就是打开持久编辑器。调用方法如下:
openPersistentEditor(QTableWidgetItem item)
该方法没有返回值。
takeItem方法是从表格部件中取一个项返回并从部件中删除该项,调用语法如下:
QTableWidgetItem takeItem(int row, int column)
项排序是指针对表格部件中的数据行进行排序,调用语法如下:
sortItems(int column, Qt.SortOrder order = Qt.AscendingOrder)
排序依据参数column指定列进行。Qt.SortOrder为枚举类,有两个常量值,分别为:AscendingOrder升序,对应数值为0,DescendingOrder为降序,对应数值为1。
项的查找和定位方法主要是查找特定的项和项的位置属性。
在表格部件中,可以根据文本以及匹配模式来搜索满足条件的项,调用语法:
list[QTableWidgetItem] findItems( str text, Qt.MatchFlags flags)
返回值为所有满足条件的项构成的列表,如果没有找到匹配项,返回空列表。
Qt.MatchFlags的取值及含义请参考《PyQt(Python+Qt)学习随笔:Model/View中的枚举类 Qt.MatchFlag的取值及含义》。
在表格部件根据选择模式的设置,只要选择模式不是NoSelection(关于选择模式继承自QAbstractItemView,请参考《PyQt(Python+Qt)学习随笔:QAbstractItemView的selectionMode属性》),则可以通过操作选中部件中的项。选中的项可以通过方法selectedItems()方法返回,其返回值为一个列表,列表中的每个元素是一个选中的QTableWidget项实例。
list[QTableWidgetItem] selectedItems()
根据行和列可以获取对应位置的项,调用语法如下:
QTableWidgetItem item(int row, int column)
如果对应位置没有项,则返回None。
QTableWidget的itemAt方法通过视口内的坐标点获取对应坐标位置的项,相关调用方法如下:
通过该方法可以获取到视口上对应坐标所在的项,如果对应坐标位置无项则返回None。
QTableWidget中可以根据项实例去定位项的行和列位置,调用方法如下:
如果对应项在QTableWidget表格部件中不存在,则返回-1。
关于逻辑行、列和可见行、列的概念,在QTableWidget中没有介绍,老猿查了比较多的资料,并经过验证,最终才搞清楚相关概念。这里有几个关键点要说明一下:
要获取表格部件中的可见行号和列号,调用方法:
QTableWidget的visualItemRect方法返回参数对应项占用视口位置的矩形数据,调用语法如下:
QRect visualItemRect( QTableWidgetItem item)
注意:
1、该方法在item对应项在视口中不可见(即需要滚动才可见)时同样会返回对应的矩形数据,只不过在视口左上角左边或上边的数据对应的横坐标或纵坐标为负数;
2、如果项对应行或列被通过表头方法hideSection隐藏了,则返回的矩形为一个空矩形(无坐标、长宽数据的矩形)。
上图中的QTableWidget部件中第一行数据已经通过hideSection进行了隐藏,第1列数据滚动到视口左边外去了,此时输出所有行第一列的可视矩形,输出结果如下:
(0,0): PyQt5.QtCore.QRect()
(1,0): PyQt5.QtCore.QRect(-76, 0, 99, 19)
(2,0): PyQt5.QtCore.QRect(-76, 20, 99, 19)
(3,0): PyQt5.QtCore.QRect(-76, 40, 99, 19)
(4,0): PyQt5.QtCore.QRect(-76, 60, 99, 19)
(5,0): PyQt5.QtCore.QRect(-76, 80, 99, 19)
(6,0): PyQt5.QtCore.QRect(-76, 100, 99, 19)
(7,0): PyQt5.QtCore.QRect(-76, 120, 99, 19)
可以看到第一行第一列因为隐藏了是个空矩形,其他行的第一列x坐标都是负数。
在QTableWidget对项的操作支持选中多个项的情况下,可以通过方法selectedRanges返回选中矩形的列表,通过方法setRangeSelected将指定矩形范围的项选中或去选中。相关方法调用语法如下:
QTableWidget表格部件的表头包括水平表头和竖直表头,水平表头每节对应表格的一列,竖直表头对应表格的一行。如图:
上图中的“行”+编号的项就是竖直表头的一个项(也称为1节),“列”+编号的项就是水平表头的一个项(节)。
setHorizontalHeaderLabels用于一次性顺序设置水平表头多个节显示的文本,调用语法如下:
setHorizontalHeaderLabels( Iterable[str] labels)
setVerticalHeaderLabels用于一次性顺序设置竖直表头多个节显示的文本,调用语法如下:
setVerticalHeaderLabels( Iterable[str] labels)
其他的方面与setHorizontalHeaderLabels方法相同。
前面介绍了,表头的一个节实际上对应一个项,项的类型与表格部件的项类型相同,都是QTableWidgetItem实例对象。水平节对应项可以通过方法horizontalHeaderItem和setHorizontalHeaderItem方法访问,调用语法如下:
类似setHorizontalHeaderItem,QTableWidget提供了verticalHeaderItem、setVerticalHeaderItem方法,调用语法如下:
相关方法使用与访问水平节类似,在此不重复说明。
QTableWidget可以取下表头节对应项并返回,相关方法如下:
对应节的项被取下后,表头对应节的标签将自动变更为节的序号。
在Designer中,QTableWidget有如下表头属性可以设置:
相关属性是从QTableView继承过来的,相关访问方法请参考《PyQt(Python+Qt)学习随笔:QTableView的标题表头相关属性》。
QTableWidget表格部件在创建以后,必须设置其包含的行数和列数才能插入项,项的插入通过setItem()进行。表格部件的QTableWidgetItem项是可用、可编辑、可选择和可选中的,并且可以用作拖放操作的源和拖放目标。项初始化后,可以进行选择、复选、编辑、排序等操作,除排序外相关操作会触发对应信号,可以通过信号来获取操作数据信息。
老猿关于PyQt的付费专栏《使用PyQt开发图形界面Python应用》只需要9.9元,该部分与第十五章的内容基本对应,但同样内容在付费专栏上总体来说更详细、案例更多。本节内容对应付费专栏的《第二十三章、 Model/View便利类表格部件QTableWidget详解》。如果有兴趣也愿意支持老猿的读者,欢迎购买付费专栏。