我在PySide2 Qt 5.13.0中创建了一个TableView
。我希望TableView
填充父项目的宽度,但是除非拖动表格,否则它不会调整列的大小。从程序开始,我希望列变宽。
main.py
import sys
from PySide2.QtCore import QUrl
from PySide2.QtQml import QQmlApplicationEngine, qmlRegisterType
from PySide2.QtWidgets import QApplication
from table_model import TableModel
import qml_rc # noqa: F401
if __name__ == "__main__":
app = QApplication(sys.argv)
engine = QQmlApplicationEngine()
qmlRegisterType(TableModel, "TableModel", 1, 0, "TableModel")
engine.load(QUrl("qrc:/main.qml"))
if not engine.rootObjects():
sys.exit(-1)
sys.exit(app.exec_())
main.qml
import QtQuick 2.13
import QtQuick.Controls 2.13
import TableModel 1.0
ApplicationWindow {
visible: true
Component.onCompleted: {
showMaximized()
}
TableView {
id: tableView
clip: true
anchors.fill: parent
model: TableModel {}
topMargin: columnsHeader.implicitHeight
columnWidthProvider: function () { return tableView.width / tableView.model.columnCount(); }
delegate: Rectangle {
implicitWidth: tableView.columnWidthProvider()
implicitHeight: 40
Text {
text: display
}
}
Row {
id: columnsHeader
y: tableView.contentY
z: 2
Repeater {
model: tableView.columns > 0 ? tableView.columns : 1
Rectangle {
width: tableView.columnWidthProvider()
height: 60
clip: true
Label {
id: headerText
width: parent.width
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
text: tableView.model.headerData(modelData, Qt.Horizontal)
elide: Text.ElideRight
clip: true
}
}
}
}
}
}
table_model.py
from typing import Any, Optional
from PySide2.QtCore import QAbstractTableModel, QModelIndex, QObject, Qt
from table import Table
class TableModel(QAbstractTableModel):
def __init__(self, parent: QObject = None) -> None:
super().__init__(parent)
self._model_data = Table(
["This", "Is", "A", "Test", "Of", "Headers"],
[
["A", 1, 2, "B", "C", "D"],
["E", 3, 4, "F", "G", "H"],
["I", 5, 6, "J", "K", "L"],
["M", 7, 8, "N", "O", "P"],
["A", 1, 2, "B", "C", "D"],
["E", 3, 4, "F", "G", "H"],
["I", 5, 6, "J", "K", "L"],
["M", 7, 8, "N", "O", "P"],
["A", 1, 2, "B", "C", "D"],
["E", 3, 4, "F", "G", "H"],
["I", 5, 6, "J", "K", "L"],
["M", 7, 8, "N", "O", "P"],
["A", 1, 2, "B", "C", "D"],
["E", 3, 4, "F", "G", "H"],
["I", 5, 6, "J", "K", "L"],
["M", 7, 8, "N", "O", "P"],
["A", 1, 2, "B", "C", "D"],
["E", 3, 4, "F", "G", "H"],
["I", 5, 6, "J", "K", "L"],
["M", 7, 8, "N", "O", "P"],
["A", 1, 2, "B", "C", "D"],
["E", 3, 4, "F", "G", "H"],
["I", 5, 6, "J", "K", "L"],
["M", 7, 8, "N", "O", "P"],
["A", 1, 2, "B", "C", "D"],
["E", 3, 4, "F", "G", "H"],
["I", 5, 6, "J", "K", "L"],
["M", 7, 8, "N", "O", "P"],
["A", 1, 2, "B", "C", "D"],
["E", 3, 4, "F", "G", "H"],
["I", 5, 6, "J", "K", "L"],
["M", 7, 8, "N", "O", "P"],
["A", 1, 2, "B", "C", "D"],
["E", 3, 4, "F", "G", "H"],
["I", 5, 6, "J", "K", "L"],
["M", 7, 8, "N", "O", "P"],
])
def rowCount(self, parent=QModelIndex()) -> int:
return len(self._model_data.rows)
def columnCount(self, parent=QModelIndex()) -> int:
return len(self._model_data.headers)
def data(self, index: QModelIndex, role=Qt.DisplayRole) -> Optional[Any]:
if role != Qt.DisplayRole:
return None
if not self.checkIndex(index, QAbstractTableModel.CheckIndexOption.IndexIsValid):
return None
return self._model_data.rows[index.row()][index.column()]
def headerData(self, section: int, orientation, role) -> Optional[str]:
if role != Qt.DisplayRole:
return None
if section < 0 or section >= len(self._model_data.headers):
return None
return self._model_data.headers[section]
def reset_with_data(self, model_data: Table) -> None:
self.beginResetModel()
self._model_data = model_data
self.endResetModel()
table.py
from dataclasses import dataclass
from typing import Any, List
@dataclass
class Table:
headers: List[str]
rows: List[List[Any]]
qml.qrc
<RCC>
<qresource prefix="/">
<file>main.qml</file>
</qresource>
</RCC>
运行:
pipenv run pyside2-rcc -o qml_rc.py qml.qrc
pipenv run python main.py
答案 0 :(得分:3)
the docs指出:
行高和列宽
当新的一栏进入视图时, TableView将通过调用columnWidthProvider确定其宽度 功能。 TableView本身永远不会存储行高或列 宽度,因为它旨在支持包含任意数量的大型模型 行和列。相反,它将在任何时候询问应用程序 需要知道。
TableView使用项目中最大的隐式宽度作为列 宽度,除非已明确设置columnWidthProvider属性。一旦 找到列宽,同一列中的所有其他项为 调整为该宽度,即使以后会轻弹新物品 具有更大的implicitWidth。在项目上设置显式宽度是 被忽略和覆盖。
注意:轻拂列时,计算出的宽度将被丢弃 在视口之外,如果将列向后滑动,则会重新计算 in。计算始终基于以下情况下可见的项目: 该列被轻拂了。这意味着它可以以不同的形式结束 时间,具体取决于列进入时所在的行。您 因此,对于一个 列,或设置columnWidthProvider。相同的逻辑适用于该行 高度计算。
如果您更改rowHeightProvider或 columnWidthProvider返回视口内的行和列, 您必须调用forceLayout。这通知TableView它需要使用 提供者将再次发挥作用,以重新计算和更新布局。
从Qt 5.13开始,如果要隐藏特定的列,则可以返回0 从该列的columnWidthProvider中。同样,您可以返回 从rowHeightProvider为0以隐藏行。如果返回负数 数字,TableView将回退以根据 委托项目。
注意:行或列的大小应为整数,以避免 项目的亚像素对齐。
以下示例显示如何设置简单的columnWidthProvider 以及用于修改函数返回值的计时器。 修改数组后,将调用forceLayout进行更改 生效:
TableView { id: tableView property var columnWidths: [100, 50, 80, 150] columnWidthProvider: function (column) { return columnWidths[column] } Timer { running: true interval: 2000 onTriggered: { tableView.columnWidths[2] = 150 tableView.forceLayout(); } } }
(重点是我的)
因此,在这种情况下,当TableView的宽度更改时,应调用forceLayout()
。就我而言(Qt 5.13.1,PySide2 5.13.1,Linux),我还必须为ApplicationWindow设置宽度和高度:
import QtQuick 2.13
import QtQuick.Controls 2.13
import TableModel 1.0
ApplicationWindow {
visible: true
Component.onCompleted: {
showMaximized()
}
width: 640
height: 480
TableView {
id: tableView
clip: true
anchors.fill: parent
model: TableModel{}
topMargin: columnsHeader.implicitHeight
columnWidthProvider: function (column) {
return tableView.model ? tableView.width/tableView.model.columnCount() : 0
}
onWidthChanged: tableView.forceLayout()
delegate: Rectangle {
implicitWidth: tableView.columnWidthProvider(column)
implicitHeight: 40
Text {
text: display
}
}
Row {
id: columnsHeader
y: tableView.contentY
z: 2
Repeater {
model: tableView.columns > 0 ? tableView.columns : 1
Rectangle {
width: tableView.columnWidthProvider(modelData)
height: 60
clip: true
Label{
id: headerText
width: parent.width
horizontalAlignment: Text.AlignHCenter
verticalAlignment: Text.AlignVCenter
text: tableView.model ? tableView.model.headerData(modelData, Qt.Horizontal): 0
elide: Text.ElideRight
clip: true
}
}
}
}
}
}