/*========================================================================= Library: CTK Copyright (c) Kitware Inc. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0.txt Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. =========================================================================*/ // Qt includes #include #include // CTK includes #include "ctkModelTester.h" //----------------------------------------------------------------------------- class ctkModelTesterPrivate { public: ctkModelTesterPrivate(); QAbstractItemModel *Model; bool ThrowOnError; bool NestedInserts; bool TestDataEnabled; bool Verbose; struct Change { QModelIndex Parent; Qt::Orientation Orientation; int Start; int End; int Count; QList Items; }; QStack AboutToBeInserted; QStack AboutToBeRemoved; QList LayoutAboutToBeChanged; }; //----------------------------------------------------------------------------- // ctkModelTesterPrivate methods //----------------------------------------------------------------------------- ctkModelTesterPrivate::ctkModelTesterPrivate() { this->Model = 0; this->ThrowOnError = true; this->NestedInserts = false; this->TestDataEnabled = true; this->Verbose = true; } //----------------------------------------------------------------------------- // ctkModelTester methods //----------------------------------------------------------------------------- ctkModelTester::ctkModelTester(QObject *_parent) :QObject(_parent) , d_ptr(new ctkModelTesterPrivate) { } //----------------------------------------------------------------------------- ctkModelTester::ctkModelTester(QAbstractItemModel *_model, QObject *_parent) :QObject(_parent) , d_ptr(new ctkModelTesterPrivate) { this->setModel(_model); } //----------------------------------------------------------------------------- ctkModelTester::~ctkModelTester() { } //----------------------------------------------------------------------------- void ctkModelTester::setModel(QAbstractItemModel *_model) { Q_D(ctkModelTester); if (d->Model) { // disconnect d->Model->disconnect(this); d->AboutToBeInserted.clear(); d->AboutToBeRemoved.clear(); d->LayoutAboutToBeChanged.clear(); } if (_model) { connect(_model, SIGNAL(columnsAboutToBeInserted(QModelIndex,int,int)), this, SLOT(onColumnsAboutToBeInserted(QModelIndex,int,int))); connect(_model, SIGNAL(columnsAboutToBeRemoved(QModelIndex,int,int)), this, SLOT(onColumnsAboutToBeRemoved(QModelIndex,int,int))); connect(_model, SIGNAL(columnsInserted(QModelIndex,int,int)), this, SLOT(onColumnsInserted(QModelIndex,int,int))); connect(_model, SIGNAL(columnsRemoved(QModelIndex,int,int)), this, SLOT(onColumnsRemoved(QModelIndex,int,int))); connect(_model, SIGNAL(dataChanged(QModelIndex,QModelIndex)), this, SLOT(onDataChanged(QModelIndex,QModelIndex))); connect(_model, SIGNAL(headerDataChanged(Qt::Orientation,int,int)), this, SLOT(onHeaderDataChanged(Qt::Orientation,int,int))); connect(_model, SIGNAL(layoutAboutToBeChanged()), this, SLOT(onLayoutAboutToBeChanged())); connect(_model, SIGNAL(layoutChanged()), this, SLOT(onLayoutChanged())); connect(_model, SIGNAL(modelAboutToBeReset()), this, SLOT(onModelAboutToBeReset())); connect(_model, SIGNAL(modelReset()), this, SLOT(onModelReset())); connect(_model, SIGNAL(rowsAboutToBeInserted(QModelIndex,int,int)), this, SLOT(onRowsAboutToBeInserted(QModelIndex,int,int))); connect(_model, SIGNAL(rowsAboutToBeRemoved(QModelIndex,int,int)), this, SLOT(onRowsAboutToBeRemoved(QModelIndex,int,int))); connect(_model, SIGNAL(rowsInserted(QModelIndex,int,int)), this, SLOT(onRowsInserted(QModelIndex,int,int))); connect(_model, SIGNAL(rowsRemoved(QModelIndex,int,int)), this, SLOT(onRowsRemoved(QModelIndex,int,int))); } d->Model = _model; this->testModel(); } //----------------------------------------------------------------------------- QAbstractItemModel* ctkModelTester::model()const { Q_D(const ctkModelTester); return d->Model; } //----------------------------------------------------------------------------- void ctkModelTester::setThrowOnError(bool throwException) { Q_D(ctkModelTester); d->ThrowOnError = throwException; } //----------------------------------------------------------------------------- bool ctkModelTester::throwOnError()const { Q_D(const ctkModelTester); return d->ThrowOnError; } //----------------------------------------------------------------------------- void ctkModelTester::setNestedInserts( bool nestedInsertsValue ) { Q_D(ctkModelTester); d->NestedInserts = nestedInsertsValue; } //----------------------------------------------------------------------------- bool ctkModelTester::nestedInserts()const { Q_D(const ctkModelTester); return d->NestedInserts; } //----------------------------------------------------------------------------- void ctkModelTester::setTestDataEnabled( bool enable ) { Q_D(ctkModelTester); d->TestDataEnabled = enable; } //----------------------------------------------------------------------------- bool ctkModelTester::testDataEnabled()const { Q_D(const ctkModelTester); return d->TestDataEnabled; } //----------------------------------------------------------------------------- void ctkModelTester::setVerbose(bool verboseValue) { Q_D(ctkModelTester); d->Verbose = verboseValue; } //----------------------------------------------------------------------------- bool ctkModelTester::verbose()const { Q_D(const ctkModelTester); return d->Verbose; } //----------------------------------------------------------------------------- void ctkModelTester::test(bool result, const QString& errorString)const { Q_D(const ctkModelTester); if (result) { return; } if (d->Verbose) { qDebug() << errorString; } if (this->throwOnError()) { throw errorString; } } //----------------------------------------------------------------------------- void ctkModelTester::testModelIndex(const QModelIndex& index)const { Q_D(const ctkModelTester); if (!index.isValid()) {// invalid index this->test(index.model() == 0, "An invalid index can't have a valid model."); this->test(index.column() == -1, "An invalid index can't have a valid column."); this->test(index.row() == -1, "An invalid index can't have a valid row."); this->test(index.parent().isValid() == false, "An invalid index can't have a valid parent."); this->test(index.row() == -1, "An invalid index can't have a valid row."); for(int i = 0; i < 100; ++i) { this->test(index.sibling(i % 10, i / 10).isValid() == false, "An invalid index can't have valid sibling."); } } else {// valid index this->test(index.model() == d->Model, "A valid index must have a valid model."); this->test(index.column() >= 0, "An valid index can't have an invalid column."); this->test(index.row() >= 0, "An valid index can't have an invalid row."); this->test(index == index.sibling(index.row(), index.column()), "Index's row and/or column is wrong."); } this->testData(index); this->testParent(index); } //----------------------------------------------------------------------------- void ctkModelTester::testData(const QModelIndex& index)const { Q_D(const ctkModelTester); if (!d->TestDataEnabled) { return; } if (!index.isValid()) { this->test(!index.data(Qt::DisplayRole).isValid(), QString("An invalid index can't have valid data: %1") .arg(index.data(Qt::DisplayRole).toString())); } else { this->test(index.data(Qt::DisplayRole).isValid(), QString("A valid index can't have invalid data: %1, %2, %3") .arg(index.row()).arg(index.column()).arg(long(index.internalPointer()))); } } //----------------------------------------------------------------------------- void ctkModelTester::testParent(const QModelIndex& vparent)const { Q_D(const ctkModelTester); if (!d->Model) { return; } if (!d->Model->hasChildren(vparent)) { // it's asking a lot :-) //this->test(d->Model->columnCount(vparent) <= 0, "A parent with no children can't have a columnCount > 0."); this->test(d->Model->rowCount(vparent) <= 0, "A parent with no children can't have a rowCount > 0."); } else { this->test(d->Model->columnCount(vparent) > 0, "A parent with children can't have a columnCount <= 0."); this->test(d->Model->rowCount(vparent) > 0 || d->Model->canFetchMore(vparent), "A parent with children can't have a rowCount <= 0. or if it does, canFetchMore should return true"); } if (!vparent.isValid()) {// otherwise there will be an infinite loop return; } for (int i = 0 ; i < d->Model->rowCount(vparent); ++i) { for (int j = 0; j < d->Model->columnCount(vparent); ++j) { this->test(d->Model->hasIndex(i, j, vparent), "hasIndex should return true for int range {0->rowCount(), 0->columnCount()}"); QModelIndex child = vparent.child(i, j); this->test(child.row() == i, "A child's row must be the same as given"); this->test(child.column() == j, "A child's column must be the same as given"); QModelIndex childParent = child.parent(); this->test(childParent == vparent, "A child's parent can't be different from its parent"); this->testModelIndex(child); } } } //----------------------------------------------------------------------------- void ctkModelTester::testPersistentModelIndex(const QPersistentModelIndex& index)const { Q_D(const ctkModelTester); //qDebug() << "Test persistent Index: " << index ; this->test(index.isValid(), "Persistent model index can't be invalid"); // did you forget to call QAbstractItemModel::changePersistentIndex() between // beginLayoutChanged() and changePersistentIndex() ? QModelIndex modelIndex = d->Model->index(index.row(), index.column(), index.parent()); this->test(modelIndex == index, QString("Persistent index (%1, %2) can't be invalid").arg(index.row()).arg(index.column())); } //----------------------------------------------------------------------------- void ctkModelTester::testModel()const { Q_D(const ctkModelTester); if (d->Model == 0) { return; } for (int i = 0 ; i < d->Model->rowCount(); ++i) { for (int j = 0; j < d->Model->columnCount(); ++j) { this->test(d->Model->hasIndex(i, j), "hasIndex should return true for int range {0->rowCount(), 0->columnCount()}"); QModelIndex child = d->Model->index(i, j); this->test(child.row() == i, "Row should be consistent"); this->test(child.column() == j, "Column should be consistent"); this->test(!child.parent().isValid(), "A child's parent can't be different from its parent"); this->testModelIndex(child); } } } //----------------------------------------------------------------------------- void ctkModelTester::onColumnsAboutToBeInserted(const QModelIndex & vparent, int start, int end) { //qDebug() << "columnsAboutToBeInserted: " << vparent << start << end; this->onItemsAboutToBeInserted(vparent, Qt::Horizontal, start, end); } //----------------------------------------------------------------------------- void ctkModelTester::onColumnsAboutToBeRemoved(const QModelIndex & vparent, int start, int end) { //qDebug() << "columnsAboutToBeRemoved: " << vparent << start << end; this->onItemsAboutToBeRemoved(vparent, Qt::Horizontal, start, end); } //----------------------------------------------------------------------------- void ctkModelTester::onColumnsInserted(const QModelIndex & vparent, int start, int end) { //qDebug() << "columnsInserted: " << vparent << start << end; this->onItemsInserted(vparent, Qt::Horizontal, start, end); } //----------------------------------------------------------------------------- void ctkModelTester::onColumnsRemoved(const QModelIndex & vparent, int start, int end) { //qDebug() << "columnsRemoved: " << vparent << start << end; this->onItemsRemoved(vparent, Qt::Horizontal, start, end); } //----------------------------------------------------------------------------- void ctkModelTester::onDataChanged(const QModelIndex & topLeft, const QModelIndex & bottomRight) { this->test(topLeft.parent() == bottomRight.parent(), "DataChanged support items with the same parent only"); this->test(topLeft.row() >= bottomRight.row(), "topLeft can't have a row lower than bottomRight"); this->test(bottomRight.column() >= topLeft.column(), "topLeft can't have a column lower than bottomRight"); for (int i = topLeft.row(); i <= bottomRight.row(); ++i) { for (int j = topLeft.column(); j <= bottomRight.column(); ++j) { this->test(topLeft.sibling(i,j).isValid(), "Changed data must be valid"); // do the test on the indexes here, it's easier to debug than in testModel(); this->testModelIndex(topLeft.sibling(i,j)); } } this->testModel(); } //----------------------------------------------------------------------------- void ctkModelTester::onHeaderDataChanged(Qt::Orientation orientation, int first, int last) { Q_D(ctkModelTester); this->test(first <= last, "Changed headers have wrong indexes"); switch (orientation) { case Qt::Horizontal: this->test(d->Model->columnCount() > last, "There is no more horizontal headers than columns."); break; case Qt::Vertical: this->test(d->Model->rowCount() > last, "There is no more vertical headers than rows."); break; } this->testModel(); } //----------------------------------------------------------------------------- QList ctkModelTester::persistentModelIndexes(const QModelIndex& index)const { Q_D(const ctkModelTester); QList list; for (int i = 0; i < d->Model->rowCount(index); ++i) { for (int j = 0; j < d->Model->columnCount(index); ++j) { QPersistentModelIndex child = d->Model->index(i, j, index); list.append(child); list += this->ctkModelTester::persistentModelIndexes(child); } } return list; } //----------------------------------------------------------------------------- void ctkModelTester::onLayoutAboutToBeChanged() { Q_D(ctkModelTester); d->LayoutAboutToBeChanged = this->persistentModelIndexes(QModelIndex()); this->testModel(); } //----------------------------------------------------------------------------- void ctkModelTester::onLayoutChanged() { Q_D(ctkModelTester); foreach (const QPersistentModelIndex& index, d->LayoutAboutToBeChanged) { this->testPersistentModelIndex(index); } d->LayoutAboutToBeChanged.clear(); this->testModel(); } //----------------------------------------------------------------------------- void ctkModelTester::onModelAboutToBeReset() { this->testModel(); } //----------------------------------------------------------------------------- void ctkModelTester::onModelReset() { this->testModel(); } //----------------------------------------------------------------------------- void ctkModelTester::onRowsAboutToBeInserted(const QModelIndex &vparent, int start, int end) { // if (d->Verbose) // { // qDebug() << "rowsAboutToBeInserted: " << vparent << start << end; // } this->onItemsAboutToBeInserted(vparent, Qt::Vertical, start, end); } //----------------------------------------------------------------------------- void ctkModelTester::onRowsAboutToBeRemoved(const QModelIndex &vparent, int start, int end) { // if (d->Verbose) // { // qDebug() << "rowsAboutToBeRemoved: " << vparent << start << end; // } this->onItemsAboutToBeRemoved(vparent, Qt::Vertical, start, end); } //----------------------------------------------------------------------------- void ctkModelTester::onRowsInserted(const QModelIndex & vparent, int start, int end) { // if (d->Verbose) // { // qDebug() << "rowsInserted: " << vparent << start << end; // } this->onItemsInserted(vparent, Qt::Vertical, start, end); } //----------------------------------------------------------------------------- void ctkModelTester::onRowsRemoved(const QModelIndex & vparent, int start, int end) { // if (d->Verbose) // { // qDebug() << "rowsRemoved: " << vparent << start << end; // } this->onItemsRemoved(vparent, Qt::Vertical, start, end); } //----------------------------------------------------------------------------- void ctkModelTester::onItemsAboutToBeInserted(const QModelIndex &vparent, Qt::Orientation orientation, int start, int end) { Q_D(ctkModelTester); this->test(start <= end, "Start can't be higher than end"); //Not sure about that if (!d->NestedInserts) { this->test(d->AboutToBeInserted.size() == 0, "While inserting items, you can't insert other items."); } //Not sure about that this->test(d->AboutToBeRemoved.size() == 0, "While removing items, you can't insert other items."); ctkModelTesterPrivate::Change change; change.Parent = vparent; change.Orientation = orientation; change.Start = start; change.End = end; change.Count = (orientation == Qt::Vertical ? d->Model->rowCount(vparent) :d->Model->columnCount(vparent) ); change.Items = this->persistentModelIndexes(vparent); d->AboutToBeInserted.push(change); this->testModel(); } //----------------------------------------------------------------------------- void ctkModelTester::onItemsAboutToBeRemoved(const QModelIndex &vparent, Qt::Orientation orientation, int start, int end) { Q_D(ctkModelTester); this->test(start <= end, "Start can't be higher than end"); //Not sure about that this->test(d->AboutToBeInserted.size() == 0, "While inserting items, you can't remove other items."); //Not sure about that this->test(d->AboutToBeRemoved.size() == 0, "While removing items, you can't remove other items."); int count = (orientation == Qt::Vertical ? d->Model->rowCount(vparent) :d->Model->columnCount(vparent) ); this->test(start < count, "Item to remove can't be invalid"); this->test(end < count, "Item to remove can't be invalid"); ctkModelTesterPrivate::Change change; change.Parent = vparent; change.Orientation = orientation; change.Start = start; change.End = end; change.Count = count; for (int i = 0 ; i < count; ++i) { QPersistentModelIndex index; index = (orientation == Qt::Vertical ? d->Model->index(i, 0, vparent) : d->Model->index(0, i, vparent)); this->test(index.isValid(), "Index invalid"); if (orientation == Qt::Vertical && (index.row() < start || index.row() > end)) { change.Items.append(index); } if (orientation == Qt::Horizontal && (index.column() < start || index.column() > end)) { change.Items.append(index); } } d->AboutToBeRemoved.push(change); this->testModel(); // if (d->Verbose) // { // qDebug() << "About to be removed: " << start << " " << end <test(start <= end, "Start can't be higher end"); this->test(d->AboutToBeInserted.size() != 0, "rowsInserted() has been emitted, but not rowsAboutToBeInserted."); //Not sure about that this->test(d->AboutToBeRemoved.size() == 0, "While removing items, you can't insert other items."); ctkModelTesterPrivate::Change change = d->AboutToBeInserted.pop(); this->test(change.Parent == vparent, "Parent can't be different"); this->test(change.Orientation == orientation, "Orientation can't be different"); this->test(change.Start == start, "Start can't be different"); this->test(change.End == end, "End can't be different"); int count = (orientation == Qt::Vertical ? d->Model->rowCount(vparent) :d->Model->columnCount(vparent) ); this->test(change.Count < count, "The new count number can't be lower"); this->test(count - change.Count == (end - start + 1) , "The new count number can't be lower"); foreach(const QPersistentModelIndex& index, change.Items) { this->testPersistentModelIndex(index); } change.Items.clear(); this->testModel(); } //----------------------------------------------------------------------------- void ctkModelTester::onItemsRemoved(const QModelIndex & vparent, Qt::Orientation orientation, int start, int end) { Q_D(ctkModelTester); this->test(start <= end, "Start can't be higher end"); this->test(d->AboutToBeRemoved.size() != 0, "rowsRemoved() has been emitted, but not rowsAboutToBeRemoved."); //Not sure about that this->test(d->AboutToBeInserted.size() == 0, "While inserted items, you can't remove other items."); ctkModelTesterPrivate::Change change = d->AboutToBeRemoved.pop(); this->test(change.Parent == vparent, "Parent can't be different"); this->test(change.Orientation == orientation, "Orientation can't be different"); this->test(change.Start == start, "Start can't be different"); this->test(change.End == end, "End can't be different"); int count = (orientation == Qt::Vertical ? d->Model->rowCount(vparent) :d->Model->columnCount(vparent) ); this->test(change.Count > count, "The new count number can't be higher"); this->test(change.Count - count == (end - start + 1) , "The new count number can't be higher"); foreach(const QPersistentModelIndex& index, change.Items) { this->testPersistentModelIndex(index); } change.Items.clear(); this->testModel(); }