ctkVTKDataSetModel.cpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345
  1. /*=========================================================================
  2. Library: CTK
  3. Copyright (c) Kitware Inc.
  4. Licensed under the Apache License, Version 2.0 (the "License");
  5. you may not use this file except in compliance with the License.
  6. You may obtain a copy of the License at
  7. http://www.commontk.org/LICENSE
  8. Unless required by applicable law or agreed to in writing, software
  9. distributed under the License is distributed on an "AS IS" BASIS,
  10. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  11. See the License for the specific language governing permissions and
  12. limitations under the License.
  13. =========================================================================*/
  14. // Qt includes
  15. #include <QDebug>
  16. // CTK includes
  17. #include "ctkVTKDataSetModel.h"
  18. // VTK includes
  19. #include <vtkCellData.h>
  20. #include <vtkSmartPointer.h>
  21. #include <vtkDataArray.h>
  22. #include <vtkDataSet.h>
  23. #include <vtkPointData.h>
  24. class ctkVTKDataSetModelPrivate
  25. {
  26. Q_DECLARE_PUBLIC(ctkVTKDataSetModel);
  27. protected:
  28. ctkVTKDataSetModel* const q_ptr;
  29. public:
  30. ctkVTKDataSetModelPrivate(ctkVTKDataSetModel& object);
  31. virtual ~ctkVTKDataSetModelPrivate();
  32. void init();
  33. //void listenDataArrayModifiedEvent();
  34. vtkSmartPointer<vtkDataSet> DataSet;
  35. bool ListenDataArrayModifiedEvent;
  36. };
  37. //------------------------------------------------------------------------------
  38. ctkVTKDataSetModelPrivate::ctkVTKDataSetModelPrivate(ctkVTKDataSetModel& object)
  39. : q_ptr(&object)
  40. {
  41. this->ListenDataArrayModifiedEvent = false;
  42. }
  43. //------------------------------------------------------------------------------
  44. ctkVTKDataSetModelPrivate::~ctkVTKDataSetModelPrivate()
  45. {
  46. }
  47. //------------------------------------------------------------------------------
  48. void ctkVTKDataSetModelPrivate::init()
  49. {
  50. Q_Q(ctkVTKDataSetModel);
  51. q->setColumnCount(1);
  52. QObject::connect(q, SIGNAL(itemChanged(QStandardItem*)),
  53. q, SLOT(onItemChanged(QStandardItem*)));
  54. }
  55. /*
  56. //------------------------------------------------------------------------------
  57. void ctkVTKDataSetModelPrivate::listenDataArrayModifiedEvent()
  58. {
  59. Q_Q(ctkVTKDataSetModel);
  60. q->qvtkDisconnect(0, vtkCommand::ModifiedEvent, q, SLOT(onArrayModified(vtkObject*)));
  61. if (!this->ListenDataArrayModifiedEvent)
  62. {
  63. return;
  64. }
  65. const int count = q->rowCount();
  66. for (int i = 0; i < count; ++i)
  67. {
  68. q->qvtkConnect(q->arrayFromIndex(q->index(i,0)),vtkCommand::ModifiedEvent,
  69. q, SLOT(onMRMLNodeModified(vtkObject*)));
  70. }
  71. }
  72. */
  73. //------------------------------------------------------------------------------
  74. // ctkVTKDataSetModel
  75. //------------------------------------------------------------------------------
  76. ctkVTKDataSetModel::ctkVTKDataSetModel(QObject *_parent)
  77. : QStandardItemModel(_parent)
  78. , d_ptr(new ctkVTKDataSetModelPrivate(*this))
  79. {
  80. Q_D(ctkVTKDataSetModel);
  81. d->init();
  82. }
  83. //------------------------------------------------------------------------------
  84. ctkVTKDataSetModel::ctkVTKDataSetModel(ctkVTKDataSetModelPrivate* pimpl, QObject *parentObject)
  85. : QStandardItemModel(parentObject)
  86. , d_ptr(pimpl)
  87. {
  88. Q_D(ctkVTKDataSetModel);
  89. d->init();
  90. }
  91. //------------------------------------------------------------------------------
  92. ctkVTKDataSetModel::~ctkVTKDataSetModel()
  93. {
  94. }
  95. //------------------------------------------------------------------------------
  96. void ctkVTKDataSetModel::setDataSet(vtkDataSet* dataSet)
  97. {
  98. Q_D(ctkVTKDataSetModel);
  99. this->qvtkReconnect(d->DataSet, dataSet, vtkCommand::ModifiedEvent,
  100. this, SLOT(onDataSetModified(vtkObject*)) );
  101. d->DataSet = dataSet;
  102. this->updateDataSet();
  103. }
  104. //------------------------------------------------------------------------------
  105. vtkDataSet* ctkVTKDataSetModel::dataSet()const
  106. {
  107. Q_D(const ctkVTKDataSetModel);
  108. return d->DataSet;
  109. }
  110. //------------------------------------------------------------------------------
  111. vtkDataArray* ctkVTKDataSetModel::arrayFromItem(QStandardItem* dataArrayItem)const
  112. {
  113. Q_D(const ctkVTKDataSetModel);
  114. if (dataArrayItem == 0 || dataArrayItem == this->invisibleRootItem())
  115. {
  116. return 0;
  117. }
  118. QVariant dataArrayPointer = dataArrayItem->data(ctkVTK::PointerRole);
  119. Q_ASSERT(dataArrayPointer.isValid());
  120. vtkDataArray* dataArray = static_cast<vtkDataArray*>(
  121. reinterpret_cast<void *>(dataArrayPointer.toLongLong()));
  122. Q_ASSERT(dataArray);
  123. return dataArray;
  124. }
  125. //------------------------------------------------------------------------------
  126. QStandardItem* ctkVTKDataSetModel::itemFromArray(vtkDataArray* dataArray, int column)const
  127. {
  128. if (dataArray == 0)
  129. {
  130. return 0;
  131. }
  132. QModelIndexList indexes = this->match(this->index(-1,-1), ctkVTK::PointerRole,
  133. reinterpret_cast<long long>(dataArray), 1,
  134. Qt::MatchExactly | Qt::MatchRecursive);
  135. while (indexes.size())
  136. {
  137. if (indexes[0].column() == column)
  138. {
  139. return this->itemFromIndex(indexes[0]);
  140. }
  141. indexes = this->match(indexes[0], ctkVTK::PointerRole,
  142. reinterpret_cast<long long>(dataArray), 1,
  143. Qt::MatchExactly | Qt::MatchRecursive);
  144. }
  145. return 0;
  146. }
  147. //------------------------------------------------------------------------------
  148. QModelIndexList ctkVTKDataSetModel::indexes(vtkDataArray* dataArray)const
  149. {
  150. return this->match(this->index(-1,-1), ctkVTK::PointerRole,
  151. QVariant::fromValue(reinterpret_cast<long long>(dataArray)),
  152. -1, Qt::MatchExactly | Qt::MatchRecursive);
  153. }
  154. /*
  155. //------------------------------------------------------------------------------
  156. void ctkVTKDataSetModel::setListenArrayModifiedEvent(bool listen)
  157. {
  158. Q_D(ctkVTKDataSetModel);
  159. if (d->ListenArrayModifiedEvent == listen)
  160. {
  161. return;
  162. }
  163. d->ListenArrayModifiedEvent = listen;
  164. d->listenArrayModifiedEvent();
  165. }
  166. //------------------------------------------------------------------------------
  167. bool ctkVTKDataSetModel::listenNodeModifiedEvent()const
  168. {
  169. Q_D(const ctkVTKDataSetModel);
  170. return d->ListenNodeModifiedEvent;
  171. }
  172. */
  173. //------------------------------------------------------------------------------
  174. void ctkVTKDataSetModel::updateDataSet()
  175. {
  176. Q_D(ctkVTKDataSetModel);
  177. this->setRowCount(0);
  178. if (d->DataSet.GetPointer() == 0)
  179. {
  180. return;
  181. }
  182. // Populate scene with nodes
  183. this->populateDataSet();
  184. }
  185. //------------------------------------------------------------------------------
  186. void ctkVTKDataSetModel::populateDataSet()
  187. {
  188. Q_D(ctkVTKDataSetModel);
  189. Q_ASSERT(d->DataSet);
  190. for (int p = 0; p < d->DataSet->GetPointData()->GetNumberOfArrays(); ++p)
  191. {
  192. this->insertArray(d->DataSet->GetPointData()->GetArray(p));
  193. }
  194. for (int p = 0; p < d->DataSet->GetCellData()->GetNumberOfArrays(); ++p)
  195. {
  196. this->insertArray(d->DataSet->GetCellData()->GetArray(p));
  197. }
  198. }
  199. //------------------------------------------------------------------------------
  200. void ctkVTKDataSetModel::insertArray(vtkDataArray* dataArray)
  201. {
  202. this->insertArray(dataArray, this->rowCount());
  203. }
  204. //------------------------------------------------------------------------------
  205. void ctkVTKDataSetModel::insertArray(vtkDataArray* dataArray, int row)
  206. {
  207. Q_D(ctkVTKDataSetModel);
  208. Q_ASSERT(vtkDataArray::SafeDownCast(dataArray));
  209. QList<QStandardItem*> items;
  210. for (int i= 0; i < this->columnCount(); ++i)
  211. {
  212. QStandardItem* newArrayItem = new QStandardItem();
  213. this->updateItemFromArray(newArrayItem, dataArray, i);
  214. items.append(newArrayItem);
  215. }
  216. this->insertRow(row,items);
  217. // TODO: don't listen to nodes that are hidden from editors ?
  218. if (d->ListenDataArrayModifiedEvent)
  219. {
  220. qvtkConnect(dataArray, vtkCommand::ModifiedEvent,
  221. this, SLOT(onArrayModified(vtkObject*)));
  222. }
  223. }
  224. //------------------------------------------------------------------------------
  225. void ctkVTKDataSetModel::updateItemFromArray(QStandardItem* item, vtkDataArray* dataArray, int column)
  226. {
  227. item->setData(QVariant::fromValue(reinterpret_cast<long long>(dataArray)), ctkVTK::PointerRole);
  228. switch (column)
  229. {
  230. case 0:
  231. item->setText(QString(dataArray->GetName()));
  232. break;
  233. default:
  234. Q_ASSERT(column == 0);
  235. break;
  236. }
  237. }
  238. //------------------------------------------------------------------------------
  239. /*
  240. void ctkVTKDataSetModel::updateItemFromPointsArray(QStandardItem* item, vtkDataArray* dataArray, int column)
  241. {
  242. this->updateItemFromArray(item, dataArray, column);
  243. switch (column)
  244. {
  245. case 0:
  246. item->setIcon();
  247. break;
  248. default:
  249. Q_ASSERT(column == 0)
  250. break;
  251. }
  252. }
  253. //------------------------------------------------------------------------------
  254. /*
  255. void ctkVTKDataSetModel::updateItemFromCellsArray(QStandardItem* item, vtkDataArray* dataArray, int column)
  256. {
  257. this->updateItemFromArray(item, dataArray, column);
  258. switch (column)
  259. {
  260. case 0:
  261. item->setIcon();
  262. break;
  263. default:
  264. Q_ASSERT(column == 0)
  265. break;
  266. }
  267. }
  268. */
  269. //------------------------------------------------------------------------------
  270. void ctkVTKDataSetModel::updateArrayFromItem(vtkDataArray* dataArray, QStandardItem* item)
  271. {
  272. if (item->column() == 0)
  273. {
  274. dataArray->SetName(item->text().toLatin1());
  275. }
  276. }
  277. //------------------------------------------------------------------------------
  278. void ctkVTKDataSetModel::onDataSetModified(vtkObject* dataSet)
  279. {
  280. this->updateDataSet();
  281. }
  282. //------------------------------------------------------------------------------
  283. void ctkVTKDataSetModel::onArrayModified(vtkObject* array)
  284. {
  285. Q_D(ctkVTKDataSetModel);
  286. vtkDataArray* dataArray = vtkDataArray::SafeDownCast(array);
  287. Q_ASSERT(dataArray);
  288. QModelIndexList arrayIndexes = this->indexes(dataArray);
  289. foreach (QModelIndex index, arrayIndexes)
  290. {
  291. QStandardItem* item = this->itemFromIndex(index);
  292. this->updateItemFromArray(item, dataArray, item->column());
  293. }
  294. }
  295. //------------------------------------------------------------------------------
  296. void ctkVTKDataSetModel::onItemChanged(QStandardItem * item)
  297. {
  298. vtkDataArray* dataArray = this->arrayFromItem(item);
  299. Q_ASSERT(dataArray);
  300. this->updateArrayFromItem(dataArray, item);
  301. }