我正在尝试实现一个可在多个列上排序的表。 Qt的QSortFilterProxyModel仅支持对一列进行排序(至少在Qt 4.6.2中)。
我在github上找到了dimkanovikov的this solution,但它缺少添加行的动态更新。我的意思是,模型被更改,并且startInsertRows(),beginRemoveRows(),它们对应的end ..-方法和dataChanged()信号被发出。理想情况下,我希望只更新这些行,但模型至少应对此类更改做出反应。
Qt的网站上有另一个FAQ项目可以对QTableWidget进行排序,但它也缺乏动态更新。
我是Qt的新手,我想知道如何解决这个问题。
答案 0 :(得分:7)
您可以将QSortFilterProxyModel
的排序角色设置为与Qt::DisplayRole
的默认setSortRole(Qt::UserRole)
不同的排序角色。然后,在模型的data()
方法中,如果使用角色Qt::UserRole
调用,则返回正确的排序键,例如通过连接相关列的字符串。
答案 1 :(得分:5)
有一个稍微不优雅的解决方案,它总是用于对多个列进行排序。
您必须继承QSortFilterProxyModel
并重新实现bool lessThan(const QModelIndex &rLeft, const QModelIndex &rRight) const
。而不是只比较两个给定的索引,检查所有列:
int const left_row = rLeft.row();
int const right_row = rRight.row();
int const num_columns = sourceModel()->columnCount();
for(int compared_column = rLeft.column(); compared_column<num_columns; ++compared_column) {
QModelIndex const left_idx = sourceModel()->index(left_row, compared_column, QModelIndex());
QModelIndex const right_idx = sourceModel()->index(right_row, compared_column, QModelIndex());
QString const leftData = sourceModel()->data(left_idx).toString();
QString const rightData = sourceModel()->data(right_idx).toString();
int const compare = QString::localeAwareCompare(leftData, rightData);
if(compare!=0) {
return compare<0;
}
}
return false;
然后,您可以在sort(0)
子类上调用QSortFilterProxyModel
,它将对所有列进行排序。当您希望在模型数据更改时动态使用已排序的行时,也不要忘记调用setDynamicSortFilter(true)
。
要支持按升序或降序对任意列进行排序,您必须将此信息保存在QList
中,并在调用lessThan
时进行相应比较。在列表中,您将按优先级顺序排列列,并按相同顺序进行比较。您还应该按照某些预定义的顺序对其他“非活动”列进行排序,否则默认情况下不会对它们进行排序。