How to sort QList<QList<QVariant>>
-
Hi,
What data are these data ?
-
Hi,
I have list of lists such QList<QList<QVariant>>, I'd like to know how one can sort on the entire list by a certain index.
So if the top level Qlist has 100 entries (say 100 rows) and each row has 10 entries each (say 10 columns), I'd like to sort the entire list by the second column. I looked around but I only see one dimensional sorts not two.
The reason I'm asking is I'm using a QAbstractTableModel with a view and when I sort the view using a proxyModel the data source is not in sync with the sort so of course the display is not what I expected. I was wondering, perhaps I just sort on the data source and it should automatically display properly on the view and even get rid of the proxyModel.
If anyone knows of a better way, please let me know or please point me in a direction to sort the data source.
Thanks!
@leinad
First you say yourQList
may have 100 rows, then you say it may have hundreds of thousands. There is a difference, especially speed-wise!If I understand, your outer
QList
is a list of rows, and your innerQList<QVariant>
is a list of the values in the columns for each row.I'm not sure which you want to achieve between just sorting the output (
QSortFiliterProxyModel
) versus actually sorting the data in the source model. But whether you re-implement[virtual]void QAbstractItemModel::sort(int column, Qt::SortOrder order = Qt::AscendingOrder
or youqsort()
the outerQList
, either way you will use the column number as index into the innerQList<QVariant>
to pick out the column value to be compared against another one. I don't see any "I looked around but I only see one dimensional sorts not two.", there is no dimensionality issue here. -
if the top level Qlist has 100 entries (say 100 rows) and each row has 10 entries each (say 10 columns), I'd like to sort the entire list by the second column
std::sort(list.begin(),list.end(),[](const QList<QVariant>& a, const QList<QVariant>& b)->bool{return a.at(1)<b.at(1);});
QAbstractTableModel with a view and when I sort the view using a proxyModel the data source is not in sync with the sort so of course the display is not what I expected
This is not correct. the
QSortFilterProxyModel
is the one that takes care of keeping a mapping between the original model and what is displayed. It should work without any problem and show exactly what you expect -
I tried using QSortFilterProxyModel but my issue is when I load thousands of rows, over time the load becomes slower and slower which is why I probabaly need to sort externally on my own. If I don't use QSortFilterProxyModel and just assign my QAbstractTableModel to my view, the load is extremely fast. So why when assigning a QSortFilterProxyModel to a view is loading the model so slow? Does the proxymodel index the data somehow?
Fast load:
view->setSourceModel(QAbstractTableModel)slow load:
proxyModel->setSourceModel(QAbstractTableModel);
view->setModel(proxyModel)Are there any flags I should set prior to load in the proxy model to speed things up?
Thanks.
-
I tried using QSortFilterProxyModel but my issue is when I load thousands of rows, over time the load becomes slower and slower which is why I probabaly need to sort externally on my own. If I don't use QSortFilterProxyModel and just assign my QAbstractTableModel to my view, the load is extremely fast. So why when assigning a QSortFilterProxyModel to a view is loading the model so slow? Does the proxymodel index the data somehow?
Fast load:
view->setSourceModel(QAbstractTableModel)slow load:
proxyModel->setSourceModel(QAbstractTableModel);
view->setModel(proxyModel)Are there any flags I should set prior to load in the proxy model to speed things up?
Thanks.
@leinad said in How to sort QList<QList<QVariant>>:
So why when assigning a QSortFilterProxyModel to a view is loading the model so slow? Does the proxymodel index the data somehow?
Yes, and it does sorting/filtering so it's slower. I cannot say whether the slowness you see at 100,000 rows is right or not, I guess sorting that takes a bit of time.
If you are happy to sort the underlying model explicitly and omit
QSortFilterProxyModel
you can do that, e.g. via thestd::sort()
@VRonin has typed in. -
Thanks. Is there anyway have it sort at the end of the load rather than as it goes? Why is it when I used to use QStandardItemModel and sorted using the treeView it was so much faster? The load was fast too. What happens under the covers using QstandardItemModel that makes such a difference?
-
Well, it looks like Vronin solution worked. I got rid of the proxyModel and attached the view directly to QAbstractTableModel and implemented the sort after the load. It loads and sort very quickly. Now I need to create a signal and slot mechanism to check when a user clicks on a particular column on the view for sorting which I will implement via std::sort.
Thanks.
-
Well, it looks like Vronin solution worked. I got rid of the proxyModel and attached the view directly to QAbstractTableModel and implemented the sort after the load. It loads and sort very quickly. Now I need to create a signal and slot mechanism to check when a user clicks on a particular column on the view for sorting which I will implement via std::sort.
Thanks.
@leinad said in How to sort QList<QList<QVariant>>:
Now I need to create a signal and slot mechanism to check when a user clicks on a particular column on the view for sorting which I will implement via std::sort.
No need to create anything fancy, just reimplement
QAbstractItemModel::sort
do do the sorting you want and everything else will be handled automatically by Qtvoid MyModel::sort(int column, Qt::SortOrder order = Qt::AscendingOrder){ emit layoutAboutToBeChanged({QModelIndex()}, QAbstractItemModel::VerticalSortHint); std::sort(list.begin(),list.end(),[column,order](const QList<QVariant>& a, const QList<QVariant>& b)->bool{ if(order == Qt::AscendingOrder) return a.at(column)<b.at(column); return a.at(column)>b.at(column); }); emit layoutChanged({QModelIndex()}, QAbstractItemModel::VerticalSortHint); }
-
@VRonin said in How to sort QList<QList<QVariant>>:
emit layoutChanged({QModelIndex()}, QAbstractItemModel::VerticalSortHint);
VRonin,
I have another question. Would you know how to handle this case in a sort? It seems to work okay when using QStandardItemModel sorting under Qt but not with the above sorting.
So the sorting works fine as is where we sort properly on column 1 which gives us the bottom results. But now within that sort I'd like to sort name1 further using column2 so name1 and name 2 is rearranged in column2 order.
If column1 is not a duplicate within the column 0 grouping then no further sort is required. So essentially take the first sort and for duplicate column 0 and column1 values resort using column 2. Is this possible?Example:
Think of column 1 as time (consider a string) where we have row data for the same object name with duplicate time, and column 2 is the rowID from a database. So even though technically the sort is correct by time the exact order in the database is governed by the rowID from a database. Essentially sort by time and then by RowID for objects in column0 and 1 that have the same value. For some reason QStandardItemModel handles this all by itself so it would be nice if I can recreate it.column 0 column1 column2
name1 123 3
name1 123 1
name1 123 2
name2 345 2
name2 345 3
name2 345 1result:
name1 123 1
name1 123 2
name1 123 3
name2 345 1
name2 345 2
name2 345 3 -
@VRonin said in How to sort QList<QList<QVariant>>:
emit layoutChanged({QModelIndex()}, QAbstractItemModel::VerticalSortHint);
VRonin,
I have another question. Would you know how to handle this case in a sort? It seems to work okay when using QStandardItemModel sorting under Qt but not with the above sorting.
So the sorting works fine as is where we sort properly on column 1 which gives us the bottom results. But now within that sort I'd like to sort name1 further using column2 so name1 and name 2 is rearranged in column2 order.
If column1 is not a duplicate within the column 0 grouping then no further sort is required. So essentially take the first sort and for duplicate column 0 and column1 values resort using column 2. Is this possible?Example:
Think of column 1 as time (consider a string) where we have row data for the same object name with duplicate time, and column 2 is the rowID from a database. So even though technically the sort is correct by time the exact order in the database is governed by the rowID from a database. Essentially sort by time and then by RowID for objects in column0 and 1 that have the same value. For some reason QStandardItemModel handles this all by itself so it would be nice if I can recreate it.column 0 column1 column2
name1 123 3
name1 123 1
name1 123 2
name2 345 2
name2 345 3
name2 345 1result:
name1 123 1
name1 123 2
name1 123 3
name2 345 1
name2 345 2
name2 345 3@leinad said in How to sort QList<QList<QVariant>>:
For some reason QStandardItemModel handles this all by itself
It doesn't. It probably just uses
std::stable_sort
instead ofstd::sort
.In any case this case is trivial, the comparison function can take care of it.
Using your example:void MyModel::sort(int column, Qt::SortOrder order = Qt::AscendingOrder){ emit layoutAboutToBeChanged({QModelIndex()}, QAbstractItemModel::VerticalSortHint); std::sort(list.begin(),list.end(),[column,order](const QList<QVariant>& a, const QList<QVariant>& b)->bool{ const auto aV = a.at(column); const auto bV = b.at(column); if(order == Qt::AscendingOrder) return aV==bV ? aV<bV : a.at(2)<b.at(2); return aV==bV ? aV>bV : a.at(2)>b.at(2); }); emit layoutChanged({QModelIndex()}, QAbstractItemModel::VerticalSortHint); }
-
@VRonin said in How to sort QList<QList<QVariant>>:
mit layoutAboutToBeChanged({QModelIndex()}, QAbstractItemModel::VerticalSortHint);
std::sort(list.begin(),list.end(),[column,order](const QList<QVariant>& a, const QList<QVariant>& b)->bool{
const auto aV = a.at(column);
const auto bV = b.at(column);
if(order == Qt::AscendingOrder)
return aV==bV ? aV<bV : a.at(2)<b.at(2);
return aV==bV ? aV>bV : a.at(2)>b.at(2);
});
emit layoutChanged({QModelIndex()}, QAbstractItemModel::VerticalSortHint);Unfortunately this does not work.
According to the code you have we don't sort on column at all. We first have to sort on column and once that is done, then we resort all duplicates of column values to some other column. -
@VRonin said in How to sort QList<QList<QVariant>>:
mit layoutAboutToBeChanged({QModelIndex()}, QAbstractItemModel::VerticalSortHint);
std::sort(list.begin(),list.end(),[column,order](const QList<QVariant>& a, const QList<QVariant>& b)->bool{
const auto aV = a.at(column);
const auto bV = b.at(column);
if(order == Qt::AscendingOrder)
return aV==bV ? aV<bV : a.at(2)<b.at(2);
return aV==bV ? aV>bV : a.at(2)>b.at(2);
});
emit layoutChanged({QModelIndex()}, QAbstractItemModel::VerticalSortHint);Unfortunately this does not work.
According to the code you have we don't sort on column at all. We first have to sort on column and once that is done, then we resort all duplicates of column values to some other column.I just inverted the 2 siders of
:
, it's trivial to fix:void MyModel::sort(int column, Qt::SortOrder order = Qt::AscendingOrder){ emit layoutAboutToBeChanged({QModelIndex()}, QAbstractItemModel::VerticalSortHint); std::sort(list.begin(),list.end(),[column,order](const QList<QVariant>& a, const QList<QVariant>& b)->bool{ const auto aV = a.at(column); const auto bV = b.at(column); if(order == Qt::AscendingOrder) return aV==bV ? a.at(2)<b.at(2) : aV<bV; return aV==bV ? a.at(2)>b.at(2) : aV>bV; }); emit layoutChanged({QModelIndex()}, QAbstractItemModel::VerticalSortHint); }