|
@@ -0,0 +1,344 @@
|
|
|
+/*=========================================================================
|
|
|
+
|
|
|
+ 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.commontk.org/LICENSE
|
|
|
+
|
|
|
+ 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 <QDebug>
|
|
|
+
|
|
|
+// CTK includes
|
|
|
+#include "ctkVTKDataSetModel.h"
|
|
|
+
|
|
|
+// VTK includes
|
|
|
+#include <vtkCellData.h>
|
|
|
+#include <vtkSmartPointer.h>
|
|
|
+#include <vtkDataArray.h>
|
|
|
+#include <vtkDataSet.h>
|
|
|
+#include <vtkPointData.h>
|
|
|
+
|
|
|
+class ctkVTKDataSetModelPrivate
|
|
|
+{
|
|
|
+ Q_DECLARE_PUBLIC(ctkVTKDataSetModel);
|
|
|
+protected:
|
|
|
+ ctkVTKDataSetModel* const q_ptr;
|
|
|
+public:
|
|
|
+ ctkVTKDataSetModelPrivate(ctkVTKDataSetModel& object);
|
|
|
+ virtual ~ctkVTKDataSetModelPrivate();
|
|
|
+ void init();
|
|
|
+ //void listenDataArrayModifiedEvent();
|
|
|
+
|
|
|
+ vtkSmartPointer<vtkDataSet> DataSet;
|
|
|
+ bool ListenDataArrayModifiedEvent;
|
|
|
+};
|
|
|
+
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+ctkVTKDataSetModelPrivate::ctkVTKDataSetModelPrivate(ctkVTKDataSetModel& object)
|
|
|
+ : q_ptr(&object)
|
|
|
+{
|
|
|
+ this->ListenDataArrayModifiedEvent = false;
|
|
|
+}
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+ctkVTKDataSetModelPrivate::~ctkVTKDataSetModelPrivate()
|
|
|
+{
|
|
|
+}
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+void ctkVTKDataSetModelPrivate::init()
|
|
|
+{
|
|
|
+ Q_Q(ctkVTKDataSetModel);
|
|
|
+ q->setColumnCount(1);
|
|
|
+
|
|
|
+ QObject::connect(q, SIGNAL(itemChanged(QStandardItem*)),
|
|
|
+ q, SLOT(onItemChanged(QStandardItem*)));
|
|
|
+}
|
|
|
+/*
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+void ctkVTKDataSetModelPrivate::listenDataArrayModifiedEvent()
|
|
|
+{
|
|
|
+ Q_Q(ctkVTKDataSetModel);
|
|
|
+ q->qvtkDisconnect(0, vtkCommand::ModifiedEvent, q, SLOT(onArrayModified(vtkObject*)));
|
|
|
+ if (!this->ListenDataArrayModifiedEvent)
|
|
|
+ {
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ const int count = q->rowCount();
|
|
|
+ for (int i = 0; i < count; ++i)
|
|
|
+ {
|
|
|
+ q->qvtkConnect(q->arrayFromIndex(q->index(i,0)),vtkCommand::ModifiedEvent,
|
|
|
+ q, SLOT(onMRMLNodeModified(vtkObject*)));
|
|
|
+ }
|
|
|
+}
|
|
|
+*/
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+// ctkVTKDataSetModel
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+ctkVTKDataSetModel::ctkVTKDataSetModel(QObject *_parent)
|
|
|
+ : QStandardItemModel(_parent)
|
|
|
+ , d_ptr(new ctkVTKDataSetModelPrivate(*this))
|
|
|
+{
|
|
|
+ Q_D(ctkVTKDataSetModel);
|
|
|
+ d->init();
|
|
|
+}
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+ctkVTKDataSetModel::ctkVTKDataSetModel(ctkVTKDataSetModelPrivate* pimpl, QObject *parentObject)
|
|
|
+ : QStandardItemModel(parentObject)
|
|
|
+ , d_ptr(pimpl)
|
|
|
+{
|
|
|
+ Q_D(ctkVTKDataSetModel);
|
|
|
+ d->init();
|
|
|
+}
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+ctkVTKDataSetModel::~ctkVTKDataSetModel()
|
|
|
+{
|
|
|
+}
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+void ctkVTKDataSetModel::setDataSet(vtkDataSet* dataSet)
|
|
|
+{
|
|
|
+ Q_D(ctkVTKDataSetModel);
|
|
|
+ this->qvtkReconnect(d->DataSet, dataSet, vtkCommand::ModifiedEvent,
|
|
|
+ this, SLOT(onDataSetModified(vtkObject*)) );
|
|
|
+ d->DataSet = dataSet;
|
|
|
+ this->updateDataSet();
|
|
|
+}
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+vtkDataSet* ctkVTKDataSetModel::dataSet()const
|
|
|
+{
|
|
|
+ Q_D(const ctkVTKDataSetModel);
|
|
|
+ return d->DataSet;
|
|
|
+}
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+vtkDataArray* ctkVTKDataSetModel::arrayFromItem(QStandardItem* dataArrayItem)const
|
|
|
+{
|
|
|
+ Q_D(const ctkVTKDataSetModel);
|
|
|
+ if (dataArrayItem == 0 || dataArrayItem == this->invisibleRootItem())
|
|
|
+ {
|
|
|
+ return 0;
|
|
|
+ }
|
|
|
+ QVariant dataArrayPointer = dataArrayItem->data(ctkVTK::PointerRole);
|
|
|
+ Q_ASSERT(dataArrayPointer.isValid());
|
|
|
+ vtkDataArray* dataArray = static_cast<vtkDataArray*>(
|
|
|
+ reinterpret_cast<void *>(dataArrayPointer.toLongLong()));
|
|
|
+ Q_ASSERT(dataArray);
|
|
|
+ return dataArray;
|
|
|
+}
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+QStandardItem* ctkVTKDataSetModel::itemFromArray(vtkDataArray* dataArray, int column)const
|
|
|
+{
|
|
|
+ if (dataArray == 0)
|
|
|
+ {
|
|
|
+ return 0;
|
|
|
+ }
|
|
|
+ QModelIndexList indexes = this->match(this->index(-1,-1), ctkVTK::PointerRole,
|
|
|
+ reinterpret_cast<long long>(dataArray), 1,
|
|
|
+ Qt::MatchExactly | Qt::MatchRecursive);
|
|
|
+ while (indexes.size())
|
|
|
+ {
|
|
|
+ if (indexes[0].column() == column)
|
|
|
+ {
|
|
|
+ return this->itemFromIndex(indexes[0]);
|
|
|
+ }
|
|
|
+ indexes = this->match(indexes[0], ctkVTK::PointerRole,
|
|
|
+ reinterpret_cast<long long>(dataArray), 1,
|
|
|
+ Qt::MatchExactly | Qt::MatchRecursive);
|
|
|
+ }
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+QModelIndexList ctkVTKDataSetModel::indexes(vtkDataArray* dataArray)const
|
|
|
+{
|
|
|
+ return this->match(this->index(-1,-1), ctkVTK::PointerRole,
|
|
|
+ QVariant::fromValue(reinterpret_cast<long long>(dataArray)),
|
|
|
+ -1, Qt::MatchExactly | Qt::MatchRecursive);
|
|
|
+}
|
|
|
+
|
|
|
+/*
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+void ctkVTKDataSetModel::setListenArrayModifiedEvent(bool listen)
|
|
|
+{
|
|
|
+ Q_D(ctkVTKDataSetModel);
|
|
|
+ if (d->ListenArrayModifiedEvent == listen)
|
|
|
+ {
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ d->ListenArrayModifiedEvent = listen;
|
|
|
+ d->listenArrayModifiedEvent();
|
|
|
+}
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+bool ctkVTKDataSetModel::listenNodeModifiedEvent()const
|
|
|
+{
|
|
|
+ Q_D(const ctkVTKDataSetModel);
|
|
|
+ return d->ListenNodeModifiedEvent;
|
|
|
+}
|
|
|
+*/
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+void ctkVTKDataSetModel::updateDataSet()
|
|
|
+{
|
|
|
+ Q_D(ctkVTKDataSetModel);
|
|
|
+ this->setRowCount(0);
|
|
|
+
|
|
|
+ if (d->DataSet.GetPointer() == 0)
|
|
|
+ {
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ // Populate scene with nodes
|
|
|
+ this->populateDataSet();
|
|
|
+}
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+void ctkVTKDataSetModel::populateDataSet()
|
|
|
+{
|
|
|
+ Q_D(ctkVTKDataSetModel);
|
|
|
+ Q_ASSERT(d->DataSet);
|
|
|
+
|
|
|
+ for (int p = 0; p < d->DataSet->GetPointData()->GetNumberOfArrays(); ++p)
|
|
|
+ {
|
|
|
+ this->insertArray(d->DataSet->GetPointData()->GetArray(p));
|
|
|
+ }
|
|
|
+
|
|
|
+ for (int p = 0; p < d->DataSet->GetCellData()->GetNumberOfArrays(); ++p)
|
|
|
+ {
|
|
|
+ this->insertArray(d->DataSet->GetCellData()->GetArray(p));
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+void ctkVTKDataSetModel::insertArray(vtkDataArray* dataArray)
|
|
|
+{
|
|
|
+ this->insertArray(dataArray, this->rowCount());
|
|
|
+}
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+void ctkVTKDataSetModel::insertArray(vtkDataArray* dataArray, int row)
|
|
|
+{
|
|
|
+ Q_D(ctkVTKDataSetModel);
|
|
|
+ Q_ASSERT(vtkDataArray::SafeDownCast(dataArray));
|
|
|
+
|
|
|
+ QList<QStandardItem*> items;
|
|
|
+ for (int i= 0; i < this->columnCount(); ++i)
|
|
|
+ {
|
|
|
+ QStandardItem* newArrayItem = new QStandardItem();
|
|
|
+ this->updateItemFromArray(newArrayItem, dataArray, i);
|
|
|
+ items.append(newArrayItem);
|
|
|
+ }
|
|
|
+ this->insertRow(row,items);
|
|
|
+ // TODO: don't listen to nodes that are hidden from editors ?
|
|
|
+ if (d->ListenDataArrayModifiedEvent)
|
|
|
+ {
|
|
|
+ qvtkConnect(dataArray, vtkCommand::ModifiedEvent,
|
|
|
+ this, SLOT(onArrayModified(vtkObject*)));
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+void ctkVTKDataSetModel::updateItemFromArray(QStandardItem* item, vtkDataArray* dataArray, int column)
|
|
|
+{
|
|
|
+ item->setData(QVariant::fromValue(reinterpret_cast<long long>(dataArray)), ctkVTK::PointerRole);
|
|
|
+ switch (column)
|
|
|
+ {
|
|
|
+ case 0:
|
|
|
+ item->setText(QString(dataArray->GetName()));
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ Q_ASSERT(column == 0);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+/*
|
|
|
+void ctkVTKDataSetModel::updateItemFromPointsArray(QStandardItem* item, vtkDataArray* dataArray, int column)
|
|
|
+{
|
|
|
+ this->updateItemFromArray(item, dataArray, column);
|
|
|
+ switch (column)
|
|
|
+ {
|
|
|
+ case 0:
|
|
|
+ item->setIcon();
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ Q_ASSERT(column == 0)
|
|
|
+ break;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+/*
|
|
|
+void ctkVTKDataSetModel::updateItemFromCellsArray(QStandardItem* item, vtkDataArray* dataArray, int column)
|
|
|
+{
|
|
|
+ this->updateItemFromArray(item, dataArray, column);
|
|
|
+ switch (column)
|
|
|
+ {
|
|
|
+ case 0:
|
|
|
+ item->setIcon();
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ Q_ASSERT(column == 0)
|
|
|
+ break;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+*/
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+void ctkVTKDataSetModel::updateArrayFromItem(vtkDataArray* dataArray, QStandardItem* item)
|
|
|
+{
|
|
|
+ if (item->column() == 0)
|
|
|
+ {
|
|
|
+ dataArray->SetName(item->text().toLatin1());
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+void ctkVTKDataSetModel::onDataSetModified(vtkObject* dataSet)
|
|
|
+{
|
|
|
+ this->updateDataSet();
|
|
|
+}
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+void ctkVTKDataSetModel::onArrayModified(vtkObject* array)
|
|
|
+{
|
|
|
+ Q_D(ctkVTKDataSetModel);
|
|
|
+ vtkDataArray* dataArray = vtkDataArray::SafeDownCast(array);
|
|
|
+ Q_ASSERT(dataArray);
|
|
|
+ QModelIndexList arrayIndexes = this->indexes(dataArray);
|
|
|
+
|
|
|
+ foreach (QModelIndex index, arrayIndexes)
|
|
|
+ {
|
|
|
+ QStandardItem* item = this->itemFromIndex(index);
|
|
|
+ this->updateItemFromArray(item, dataArray, item->column());
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+//------------------------------------------------------------------------------
|
|
|
+void ctkVTKDataSetModel::onItemChanged(QStandardItem * item)
|
|
|
+{
|
|
|
+ vtkDataArray* dataArray = this->arrayFromItem(item);
|
|
|
+ Q_ASSERT(dataArray);
|
|
|
+ this->updateArrayFromItem(dataArray, item);
|
|
|
+}
|