diff options
author | Stephen Kelly <stephen.kelly@kdab.com> | 2012-08-13 11:53:05 +0200 |
---|---|---|
committer | Qt by Nokia <qt-info@nokia.com> | 2012-08-18 08:34:53 +0200 |
commit | 2ef3ac72fc068b825f2185ab3682219dc7246464 (patch) | |
tree | ef08dbea5c53c950a22fe8752ec4b2f0f3d990ab | |
parent | 72604f8eb9e0cab8e0de8f30576af11aebaf3e11 (diff) | |
download | qtbase-2ef3ac72fc068b825f2185ab3682219dc7246464.tar.gz |
Emit layoutChange signals when changing QPersistentModelIndexes.
This is necessary whenever QPersistentModelIndexes are changed. Omitting
it means that views are not able to react to the change, such as QTreeView
clearing its (manually held) QModelIndex cache, and the QItemSelectionModel
clearing the item from its storage.
It is necessary to change a QSortFilterProxyModel test which assumed setItem
does not have any such effect. That test is ported to setData instead.
Task-number: QTBUG-18539
Change-Id: Id7a602f18b9773ba4d11019418de886860d26d3e
Reviewed-by: Marc Mutz <marc.mutz@kdab.com>
Reviewed-by: Stephen Kelly <stephen.kelly@kdab.com>
3 files changed, 49 insertions, 1 deletions
diff --git a/src/gui/itemmodels/qstandarditemmodel.cpp b/src/gui/itemmodels/qstandarditemmodel.cpp index 2d296a0e59..223c34a517 100644 --- a/src/gui/itemmodels/qstandarditemmodel.cpp +++ b/src/gui/itemmodels/qstandarditemmodel.cpp @@ -139,6 +139,11 @@ void QStandardItemPrivate::setChild(int row, int column, QStandardItem *item, QStandardItem *oldItem = children.at(index); if (item == oldItem) return; + + if (model && emitChanged) { + emit model->layoutAboutToBeChanged(); + } + if (item) { if (item->d_func()->parent == 0) { item->d_func()->setParentAndModel(q, model); @@ -152,6 +157,10 @@ void QStandardItemPrivate::setChild(int row, int column, QStandardItem *item, oldItem->d_func()->setModel(0); delete oldItem; children.replace(index, item); + + if (model && emitChanged) + emit model->layoutChanged(); + if (emitChanged && model) model->d_func()->itemChanged(item); } diff --git a/tests/auto/corelib/itemmodels/qsortfilterproxymodel/tst_qsortfilterproxymodel.cpp b/tests/auto/corelib/itemmodels/qsortfilterproxymodel/tst_qsortfilterproxymodel.cpp index 444d26ce33..7f141bdab0 100644 --- a/tests/auto/corelib/itemmodels/qsortfilterproxymodel/tst_qsortfilterproxymodel.cpp +++ b/tests/auto/corelib/itemmodels/qsortfilterproxymodel/tst_qsortfilterproxymodel.cpp @@ -2496,7 +2496,7 @@ void tst_QSortFilterProxyModel::staticSorting() } //update one item. - model.setItem(0, 0, new QStandardItem("girafe")); + items.first()->setData("girafe", Qt::DisplayRole); // make sure the proxy is updated but not sorted expected.replaceInStrings("bateau", "girafe"); diff --git a/tests/auto/widgets/itemviews/qtreeview/tst_qtreeview.cpp b/tests/auto/widgets/itemviews/qtreeview/tst_qtreeview.cpp index bbb51aed5e..84acac5a96 100644 --- a/tests/auto/widgets/itemviews/qtreeview/tst_qtreeview.cpp +++ b/tests/auto/widgets/itemviews/qtreeview/tst_qtreeview.cpp @@ -240,6 +240,7 @@ private slots: void taskQTBUG_11466_keyboardNavigationRegression(); void taskQTBUG_13567_removeLastItemRegression(); void taskQTBUG_25333_adjustViewOptionsForIndex(); + void taskQTBUG_18539_emitLayoutChanged(); }; class QtTestModel: public QAbstractItemModel @@ -4054,6 +4055,44 @@ void tst_QTreeView::taskQTBUG_25333_adjustViewOptionsForIndex() } +void tst_QTreeView::taskQTBUG_18539_emitLayoutChanged() +{ + QTreeView view; + + QStandardItem* item = new QStandardItem("Orig"); + QStandardItem* child = new QStandardItem("Child"); + item->setChild(0, 0, child); + + QStandardItemModel model; + model.appendRow(item); + + view.setModel(&model); + + QStandardItem* replacementItem = new QStandardItem("Replacement"); + QStandardItem* replacementChild = new QStandardItem("ReplacementChild"); + + replacementItem->setChild(0, 0, replacementChild); + + QSignalSpy beforeSpy(&model, SIGNAL(layoutAboutToBeChanged())); + QSignalSpy afterSpy(&model, SIGNAL(layoutChanged())); + + QSignalSpy beforeRISpy(&model, SIGNAL(rowsAboutToBeInserted(QModelIndex,int,int))); + QSignalSpy afterRISpy(&model, SIGNAL(rowsInserted(QModelIndex,int,int))); + + QSignalSpy beforeRRSpy(&model, SIGNAL(rowsAboutToBeRemoved(QModelIndex,int,int))); + QSignalSpy afterRRSpy(&model, SIGNAL(rowsRemoved(QModelIndex,int,int))); + + model.setItem(0, 0, replacementItem); + + QCOMPARE(beforeSpy.size(), 1); + QCOMPARE(afterSpy.size(), 1); + + QCOMPARE(beforeRISpy.size(), 0); + QCOMPARE(afterRISpy.size(), 0); + + QCOMPARE(beforeRISpy.size(), 0); + QCOMPARE(afterRISpy.size(), 0); +} QTEST_MAIN(tst_QTreeView) #include "tst_qtreeview.moc" |