【发布时间】:2021-08-28 15:47:50
【问题描述】:
我正在尝试使用beginMoveColumns 在QTableView 中移动单个列,但在下面的示例中它不能正常工作。单元格选择被打乱,列宽不移动。使用相同逻辑移动行似乎可以正常工作。我做错了什么?
视频:https://www.screencast.com/t/5UJ0iByZCEE
from PyQt5 import QtWidgets, QtCore, QtGui
import sys
from PyQt5.QtCore import QModelIndex, Qt
class MyTableModel(QtCore.QAbstractTableModel):
def __init__(self, data=[[]], parent=None):
super().__init__(parent)
self.data = data
def headerData(self, section: int, orientation: Qt.Orientation, role: int):
if role == QtCore.Qt.DisplayRole:
return "Header"
def columnCount(self, parent=None):
return len(self.data[0])
def rowCount(self, parent=None):
return len(self.data)
def data(self, index: QModelIndex, role: int):
if role == QtCore.Qt.DisplayRole:
row = index.row()
col = index.column()
return str(self.data[row][col])
def move_column(self, ix, new_ix):
parent = QtCore.QModelIndex()
if new_ix > ix:
target = new_ix + 1
else:
target = new_ix
self.beginMoveColumns(parent, ix, ix, parent, target)
# Shift column in each row
for row in self.data:
row.insert(new_ix, row.pop(ix))
self.endMoveColumns()
def move_row(self, ix, new_ix):
parent = QtCore.QModelIndex()
if new_ix > ix:
target = new_ix + 1
else:
target = new_ix
self.beginMoveRows(parent, ix, ix, parent, target)
# Shift row
self.data.insert(new_ix, self.data.pop(ix))
self.endMoveRows()
if __name__ == '__main__':
app = QtWidgets.QApplication(sys.argv)
data = []
for i in range(12):
row = []
data.append(row)
for j in range(6):
row.append(f"R{i}_C{j}")
model = MyTableModel(data)
view = QtWidgets.QTableView()
view.setModel(model)
view.setColumnWidth(0, 50)
view.setColumnWidth(1, 100)
view.setRowHeight(0, 50)
view.setRowHeight(1, 100)
container = QtWidgets.QWidget()
layout = QtWidgets.QVBoxLayout()
container.setLayout(layout)
layout.addWidget(view)
button = QtWidgets.QPushButton("Move 1st column right by 1")
button.clicked.connect(lambda: model.move_column(0, 1))
layout.addWidget(button)
button = QtWidgets.QPushButton("Move 1st column right by 2")
button.clicked.connect(lambda: model.move_column(0, 2))
layout.addWidget(button)
button = QtWidgets.QPushButton("Move 1st row down by 1")
button.clicked.connect(lambda: model.move_row(0, 1))
layout.addWidget(button)
button = QtWidgets.QPushButton("Move 1st row down by 2")
button.clicked.connect(lambda: model.move_row(0, 2))
layout.addWidget(button)
container.resize(800, 800)
container.show()
sys.exit(app.exec_())
【问题讨论】:
-
我试图了解奇怪的选择行为,因为它可能是一个错误。也就是说,标题的行为是预期的:移动行或列不会更改行和列的索引(因此,标题部分):您正在移动内部数据,而不是“实际”行/列索引.如果您需要移动显示的顺序,那么您不应该更改数据模型,而是移动标题部分(使用
header.moveSection())。 -
嗯,我的误解可能比我想象的要多。 QAtbstractTableModel 不应该也处理移动标题数据吗?我的目标是完全移动列和行(单元格数据和标题数据)。这种移动应该是对底层数据的修改,应该反映在视图中,并且我希望列和行大小跟随数据。这是我使用 Excel 执行我描述的操作的片段screencast.com/t/uajgleiH
-
另一件重要的事情(与上一篇一样对您的原始帖子有效),您不应将数据引用命名为
self.data:data()是访问数据模型所调用的方法(并且可以通过model.data()访问),所以它应该不被覆盖。 -
带有“...不是标准行为” 我指的是移动基础数据行/列时 headers 的移动。虽然我可以理解您的观点,但这不是 Qt 项目模型的基本实现的工作方式。您可以通过在移动行/列时重新排序标题列表来提供该功能,类似于
reindex所做的。关于这一点:由于数据帧返回的索引的类型,标题变得不可见;重新索引会导致标题变成int64类型,而 PyQt 对这种类型一无所知:使用int()来解决这个问题。
标签: python pyqt qtableview qabstracttablemodel