implementing sorting with QSortFilterProxyModel
-
@Christian-Ehrlicher ahhhhhh...thank you for the emphasis; now I understand what you're saying. But...I don't know how to do that.
My view is a QML ListView. How does one effect a sort on that?
@mzimmers
I would just start by GooglingQML QSortFilterProxyModel
. https://stackoverflow.com/questions/45563393/sort-the-data-of-a-qml-listview-by-using-a-qsortfilterproxymodel might be just what you need to understand from QML (though it might be too detailed), I don't know. Note that it talks about implementing thelessThan()
. That is all you have to do for aQSortFilterProxyModel
. Its own code will call that repeatedly in order to present its source model in sorted order to the outside world. -
Hi all -
I've implemented a QSortFilterProxyModel (on a QAbstractListModel) for the purpose of filtering, which it does just fine. Now, I want to implement a sort of the filtered data. It's not clear to me from the docs how I cause this to happen. The docs state:
Custom sorting behavior is achieved by subclassing QSortFilterProxyModel and reimplementing lessThan(), which is used to compare items.
I've done this, but my lessThan() is never called.What do I need to do in order to "activate" sorting? From the examples, it doesn't appear that I should actually invoke the sort() method directly.
EDIT:
Here's the signatures of my reimplmented functions:
protected: bool filterAcceptsRow(int sourceRow, const QModelIndex &sourceParent) const override; bool lessThan(const QModelIndex &left, const QModelIndex &right) const override;
Thanks...
@mzimmers Here is an exemple.Hope it helps
SortOnSecondModel.hclass SortOnSecondModel : public QSortFilterProxyModel
{
public:
SortOnSecondModel( QObject *parent = 0 );
protected:
bool lessThan( const QModelIndex &left, const QModelIndex &right ) const;
};SortOnSecondModel.cpp
bool SortOnSecondModel::lessThan( const QModelIndex &left,
const QModelIndex &right ) const
{
QString leftString = sourceModel()->data( left ).toString();
QString rightString = sourceModel()->data( right ).toString();
if( !leftString.isEmpty() )
leftString = leftString.mid( 1 );
if( !rightString.isEmpty() )
rightString = rightstring.mid( 1 );
return leftString < rightString;
} -
@mzimmers Here is an exemple.Hope it helps
SortOnSecondModel.hclass SortOnSecondModel : public QSortFilterProxyModel
{
public:
SortOnSecondModel( QObject *parent = 0 );
protected:
bool lessThan( const QModelIndex &left, const QModelIndex &right ) const;
};SortOnSecondModel.cpp
bool SortOnSecondModel::lessThan( const QModelIndex &left,
const QModelIndex &right ) const
{
QString leftString = sourceModel()->data( left ).toString();
QString rightString = sourceModel()->data( right ).toString();
if( !leftString.isEmpty() )
leftString = leftString.mid( 1 );
if( !rightString.isEmpty() )
rightString = rightstring.mid( 1 );
return leftString < rightString;
}@Ronel_qtmaster thanks for the example. I haven't gotten around to implementing my lessThan() function, because I'm still trying to figure out how/where to invoke the sort() call. This is what I'm trying:
EquipmentValveProxy::EquipmentValveProxy(EquipmentModel *equipmentModel, QObject *parent) : QSortFilterProxyModel{parent}, m_equipmentModel(equipmentModel) { QObject::connect(m_equipmentModel, &EquipmentModel::modelUpdated, this, &EquipmentValveProxy::updateSorting); } void EquipmentValveProxy::updateSorting() { invalidate(); sort(0); // what should I use for a column? }
This is successful in that it gets me into my lessThan() function, currently stubbed as:
bool EquipmentValveProxy::lessThan(const QModelIndex &left, const QModelIndex &right) const { bool rc = false; QVariant leftData = sourceModel()->data(left); QVariant rightData = sourceModel()->data(right); return rc; }
Unfortunately, I'm still doing something wrong, because both leftData and rightData are invalid after assignments. How does this look to you?
Thanks...
-
@Ronel_qtmaster thanks for the example. I haven't gotten around to implementing my lessThan() function, because I'm still trying to figure out how/where to invoke the sort() call. This is what I'm trying:
EquipmentValveProxy::EquipmentValveProxy(EquipmentModel *equipmentModel, QObject *parent) : QSortFilterProxyModel{parent}, m_equipmentModel(equipmentModel) { QObject::connect(m_equipmentModel, &EquipmentModel::modelUpdated, this, &EquipmentValveProxy::updateSorting); } void EquipmentValveProxy::updateSorting() { invalidate(); sort(0); // what should I use for a column? }
This is successful in that it gets me into my lessThan() function, currently stubbed as:
bool EquipmentValveProxy::lessThan(const QModelIndex &left, const QModelIndex &right) const { bool rc = false; QVariant leftData = sourceModel()->data(left); QVariant rightData = sourceModel()->data(right); return rc; }
Unfortunately, I'm still doing something wrong, because both leftData and rightData are invalid after assignments. How does this look to you?
Thanks...
@mzimmers said in implementing sorting with QSortFilterProxyModel:
To invoque the sort , you just need to call QTableView::setSortingEnabled( true );
In your EquipmentValveProxy class, you don't need any connect in the constructor, you only need to reimplement lessThan function.
Now you need to pass the model containing your data to the sorter class.
EquipmentValveProxy ::setSourceModel( &model );Then set the sorter class as model to your QTableView
QTableView::setModel(&your_sorter_class);And finally enable sorting using QTableView::setSortingEnabled( true );
-
@mzimmers said in implementing sorting with QSortFilterProxyModel:
To invoque the sort , you just need to call QTableView::setSortingEnabled( true );
In your EquipmentValveProxy class, you don't need any connect in the constructor, you only need to reimplement lessThan function.
Now you need to pass the model containing your data to the sorter class.
EquipmentValveProxy ::setSourceModel( &model );Then set the sorter class as model to your QTableView
QTableView::setModel(&your_sorter_class);And finally enable sorting using QTableView::setSortingEnabled( true );
@Ronel_qtmaster said in implementing sorting with QSortFilterProxyModel:
To invoque the sort , you just need to call QTableView::setSortingEnabled( true );
But...it's a QML ListView, not a C++ TableView.
-
@Ronel_qtmaster said in implementing sorting with QSortFilterProxyModel:
To invoque the sort , you just need to call QTableView::setSortingEnabled( true );
But...it's a QML ListView, not a C++ TableView.
@mzimmers Okay.Then you should consider porting your model to qml https://doc.qt.io/qt-6/qtquick-modelviewsdata-cppmodels.html
-
@mzimmers Okay.Then you should consider porting your model to qml https://doc.qt.io/qt-6/qtquick-modelviewsdata-cppmodels.html
@Ronel_qtmaster said in implementing sorting with QSortFilterProxyModel:
you should consider porting your model to qml
I assume you mean making my model available to QML? I've done that. I've gone through (most of) the steps in the video from the link you provided.
If you mean actually convert my model to a QML ListModel, that's not an option. My model code is ~1000 lines of C++ and fairly intricate; it's not conducive to conversion to QML.
What I'm doing (with the connection) seems to be working. In the debugger, it's alarmingly slow, but when run normally, performance seems OK. Unless there's a serious flaw in this approach, I think it's what I have to do.
-
@Ronel_qtmaster thanks for the example. I haven't gotten around to implementing my lessThan() function, because I'm still trying to figure out how/where to invoke the sort() call. This is what I'm trying:
EquipmentValveProxy::EquipmentValveProxy(EquipmentModel *equipmentModel, QObject *parent) : QSortFilterProxyModel{parent}, m_equipmentModel(equipmentModel) { QObject::connect(m_equipmentModel, &EquipmentModel::modelUpdated, this, &EquipmentValveProxy::updateSorting); } void EquipmentValveProxy::updateSorting() { invalidate(); sort(0); // what should I use for a column? }
This is successful in that it gets me into my lessThan() function, currently stubbed as:
bool EquipmentValveProxy::lessThan(const QModelIndex &left, const QModelIndex &right) const { bool rc = false; QVariant leftData = sourceModel()->data(left); QVariant rightData = sourceModel()->data(right); return rc; }
Unfortunately, I'm still doing something wrong, because both leftData and rightData are invalid after assignments. How does this look to you?
Thanks...
@mzimmers said in implementing sorting with QSortFilterProxyModel:
sort(0); // what should I use for a column?
Not sure what your question is here? Only you know which column you want to sort by. If you go back to that full example I referenced on SO you will see they do indeed use
m_messageProxyModel.sort(0, Qt::AscendingOrder);
so that's the right thing to do.The
QTableView
widget has asetSortingEnabled()
. If set that enables "sort arrows" on each column header so user can click to sort by any column. The view invokes theQSFPM::sort()
method with corresponding column/direction. If that is what you want (end-user sorting) does your QML "grid" or "table" or whatever offer this? Otherwise have a look at the accepted solution in that SO thread:Alternatively, and if you want to do the sorting and filtering from QML and not c++, you could use my SortFilterProxyModel like so
So that guy has written a complete QML interface to doing sort/filtering for you just like
QSFPM
for widgets! Don't look a gift horse in the mouth. (Don't actually know what you're supposed to do with one, because if you take it in you get invaded, so just looking at it in the mouth seems as safe as anything to me...) Either take the whole thing or look up how he does relevant bits? You did look at this when I first gave you the reference, didn't you? ;-)Unfortunately, I'm still doing something wrong, because both leftData and rightData are invalid after assignments. How does this look to you?
What do you mean "invalid after assignments"? That should correctly pick up the
QVariant
values at the model indexes, whatever those value are. -
@mzimmers said in implementing sorting with QSortFilterProxyModel:
sort(0); // what should I use for a column?
Not sure what your question is here? Only you know which column you want to sort by. If you go back to that full example I referenced on SO you will see they do indeed use
m_messageProxyModel.sort(0, Qt::AscendingOrder);
so that's the right thing to do.The
QTableView
widget has asetSortingEnabled()
. If set that enables "sort arrows" on each column header so user can click to sort by any column. The view invokes theQSFPM::sort()
method with corresponding column/direction. If that is what you want (end-user sorting) does your QML "grid" or "table" or whatever offer this? Otherwise have a look at the accepted solution in that SO thread:Alternatively, and if you want to do the sorting and filtering from QML and not c++, you could use my SortFilterProxyModel like so
So that guy has written a complete QML interface to doing sort/filtering for you just like
QSFPM
for widgets! Don't look a gift horse in the mouth. (Don't actually know what you're supposed to do with one, because if you take it in you get invaded, so just looking at it in the mouth seems as safe as anything to me...) Either take the whole thing or look up how he does relevant bits? You did look at this when I first gave you the reference, didn't you? ;-)Unfortunately, I'm still doing something wrong, because both leftData and rightData are invalid after assignments. How does this look to you?
What do you mean "invalid after assignments"? That should correctly pick up the
QVariant
values at the model indexes, whatever those value are.@JonB said in implementing sorting with QSortFilterProxyModel:
If you go back to that full example I referenced on SO you will see they do indeed use m_messageProxyModel.sort(0, Qt::AscendingOrder); so that's the right thing to do.
Yeah, I figured out that wasn't the problem a little after I posted.
The reason that my leftData and rightData variables were invalid were simply that I hadn't furnished a role from my model. I changed the lines to this:
QVariant leftData = sourceModel()->data(left, EquipmentModel::NameRole); QVariant rightData = sourceModel()->data(right, EquipmentModel::NameRole);
And now it works. Incredibly slow in the debugger, but otherwise fine.
And yes, I did look at @GrecKo 's solution that he referenced on SO. If my needs become more elaborate than they currently are, I'll reconsider that option.
-
M mzimmers has marked this topic as solved on
-
@Ronel_qtmaster said in implementing sorting with QSortFilterProxyModel:
you should consider porting your model to qml
I assume you mean making my model available to QML? I've done that. I've gone through (most of) the steps in the video from the link you provided.
If you mean actually convert my model to a QML ListModel, that's not an option. My model code is ~1000 lines of C++ and fairly intricate; it's not conducive to conversion to QML.
What I'm doing (with the connection) seems to be working. In the debugger, it's alarmingly slow, but when run normally, performance seems OK. Unless there's a serious flaw in this approach, I think it's what I have to do.
@mzimmers If your model is to be consumed in QML, it most likely has a single column and multiple roles if it's meant to be consumed by a ListView (multiple columns shouldn't be a thing unless you are implementing a spreadsheet, change my mind).
call
setDynamicSortFilter(true)
andsort(0)
once at the beginning and you should be good to go. YourlessThan
function will be called when needed. Or use my project on GitHub like mentioned above. It hasn't been updated for a while but it is still relevant. Take a look at the available forks too, there might be improvements there. -
@mzimmers If your model is to be consumed in QML, it most likely has a single column and multiple roles if it's meant to be consumed by a ListView (multiple columns shouldn't be a thing unless you are implementing a spreadsheet, change my mind).
call
setDynamicSortFilter(true)
andsort(0)
once at the beginning and you should be good to go. YourlessThan
function will be called when needed. Or use my project on GitHub like mentioned above. It hasn't been updated for a while but it is still relevant. Take a look at the available forks too, there might be improvements there. -
M mzimmers has marked this topic as unsolved on
-
M mzimmers has marked this topic as solved on
-
@GrecKo said in implementing sorting with QSortFilterProxyModel:
call setDynamicSortFilter(true) and sort(0) once at the beginning and you should be good to go.
Is this true even if the source model adds or deletes list elements?
It is.
-