ctkVTKDataSetModel.cpp 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408
  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.apache.org/licenses/LICENSE-2.0.txt
  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. static QList<vtkDataArray*> attributeArrayToInsert(const ctkVTKDataSetModel::AttributeTypes& attributeType,
  35. vtkDataSetAttributes * dataSetAttributes);
  36. vtkSmartPointer<vtkDataSet> DataSet;
  37. bool ListenDataArrayModifiedEvent;
  38. ctkVTKDataSetModel::AttributeTypes AttributeType;
  39. };
  40. //------------------------------------------------------------------------------
  41. ctkVTKDataSetModelPrivate::ctkVTKDataSetModelPrivate(ctkVTKDataSetModel& object)
  42. : q_ptr(&object)
  43. {
  44. this->ListenDataArrayModifiedEvent = false;
  45. this->AttributeType = ctkVTKDataSetModel::AllAttribute;
  46. }
  47. //------------------------------------------------------------------------------
  48. ctkVTKDataSetModelPrivate::~ctkVTKDataSetModelPrivate()
  49. {
  50. }
  51. //------------------------------------------------------------------------------
  52. void ctkVTKDataSetModelPrivate::init()
  53. {
  54. Q_Q(ctkVTKDataSetModel);
  55. q->setColumnCount(1);
  56. QObject::connect(q, SIGNAL(itemChanged(QStandardItem*)),
  57. q, SLOT(onItemChanged(QStandardItem*)));
  58. }
  59. /*
  60. //------------------------------------------------------------------------------
  61. void ctkVTKDataSetModelPrivate::listenDataArrayModifiedEvent()
  62. {
  63. Q_Q(ctkVTKDataSetModel);
  64. q->qvtkDisconnect(0, vtkCommand::ModifiedEvent, q, SLOT(onArrayModified(vtkObject*)));
  65. if (!this->ListenDataArrayModifiedEvent)
  66. {
  67. return;
  68. }
  69. const int count = q->rowCount();
  70. for (int i = 0; i < count; ++i)
  71. {
  72. q->qvtkConnect(q->arrayFromIndex(q->index(i,0)),vtkCommand::ModifiedEvent,
  73. q, SLOT(onMRMLNodeModified(vtkObject*)));
  74. }
  75. }
  76. */
  77. //------------------------------------------------------------------------------
  78. QList<vtkDataArray*> ctkVTKDataSetModelPrivate::attributeArrayToInsert(
  79. const ctkVTKDataSetModel::AttributeTypes& attributeType,
  80. vtkDataSetAttributes * dataSetAttributes)
  81. {
  82. QList<vtkDataArray*> attributeArraysToInsert;
  83. for (int p = 0; p < dataSetAttributes->GetNumberOfArrays(); ++p)
  84. {
  85. vtkDataArray * dataArray = dataSetAttributes->GetArray(p);
  86. bool isAttributeArray = false;
  87. vtkDataArray* attributeArrays[vtkDataSetAttributes::NUM_ATTRIBUTES];
  88. for(int attributeId = 0; attributeId < vtkDataSetAttributes::NUM_ATTRIBUTES; ++attributeId)
  89. {
  90. attributeArrays[attributeId] = dataSetAttributes->GetAttribute(attributeId);
  91. if (!isAttributeArray && attributeArrays[attributeId] == dataArray)
  92. {
  93. isAttributeArray = true;
  94. }
  95. }
  96. if ((attributeType & ctkVTKDataSetModel::ScalarsAttribute && (dataArray == attributeArrays[vtkDataSetAttributes::SCALARS]))
  97. || (attributeType & ctkVTKDataSetModel::VectorsAttribute && (dataArray == attributeArrays[vtkDataSetAttributes::VECTORS]))
  98. || (attributeType & ctkVTKDataSetModel::NormalsAttribute && (dataArray == attributeArrays[vtkDataSetAttributes::NORMALS]))
  99. || (attributeType & ctkVTKDataSetModel::TCoordsAttribute && (dataArray == attributeArrays[vtkDataSetAttributes::TCOORDS]))
  100. || (attributeType & ctkVTKDataSetModel::TensorsAttribute && (dataArray == attributeArrays[vtkDataSetAttributes::TENSORS]))
  101. || (attributeType & ctkVTKDataSetModel::GlobalIDsAttribute && (dataArray == attributeArrays[vtkDataSetAttributes::GLOBALIDS]))
  102. || (attributeType & ctkVTKDataSetModel::PedigreeIDsAttribute && (dataArray == attributeArrays[vtkDataSetAttributes::PEDIGREEIDS]))
  103. || (attributeType & ctkVTKDataSetModel::EdgeFlagAttribute && (dataArray == attributeArrays[vtkDataSetAttributes::EDGEFLAG]))
  104. || (attributeType & ctkVTKDataSetModel::NoAttribute && !isAttributeArray)
  105. )
  106. {
  107. attributeArraysToInsert << dataSetAttributes->GetArray(p);
  108. }
  109. }
  110. return attributeArraysToInsert;
  111. }
  112. //------------------------------------------------------------------------------
  113. // ctkVTKDataSetModel
  114. //------------------------------------------------------------------------------
  115. ctkVTKDataSetModel::ctkVTKDataSetModel(QObject *_parent)
  116. : QStandardItemModel(_parent)
  117. , d_ptr(new ctkVTKDataSetModelPrivate(*this))
  118. {
  119. Q_D(ctkVTKDataSetModel);
  120. d->init();
  121. }
  122. //------------------------------------------------------------------------------
  123. ctkVTKDataSetModel::ctkVTKDataSetModel(ctkVTKDataSetModelPrivate* pimpl, QObject *parentObject)
  124. : QStandardItemModel(parentObject)
  125. , d_ptr(pimpl)
  126. {
  127. Q_D(ctkVTKDataSetModel);
  128. d->init();
  129. }
  130. //------------------------------------------------------------------------------
  131. ctkVTKDataSetModel::~ctkVTKDataSetModel()
  132. {
  133. }
  134. //------------------------------------------------------------------------------
  135. void ctkVTKDataSetModel::setDataSet(vtkDataSet* dataSet)
  136. {
  137. Q_D(ctkVTKDataSetModel);
  138. if (dataSet == d->DataSet.GetPointer())
  139. {
  140. return;
  141. }
  142. this->qvtkReconnect(d->DataSet, dataSet, vtkCommand::ModifiedEvent,
  143. this, SLOT(onDataSetModified(vtkObject*)) );
  144. d->DataSet = dataSet;
  145. this->updateDataSet();
  146. }
  147. //------------------------------------------------------------------------------
  148. vtkDataSet* ctkVTKDataSetModel::dataSet()const
  149. {
  150. Q_D(const ctkVTKDataSetModel);
  151. return d->DataSet;
  152. }
  153. //------------------------------------------------------------------------------
  154. ctkVTKDataSetModel::AttributeTypes ctkVTKDataSetModel::attributeTypes()const
  155. {
  156. Q_D(const ctkVTKDataSetModel);
  157. return d->AttributeType;
  158. }
  159. //------------------------------------------------------------------------------
  160. void ctkVTKDataSetModel::setAttributeTypes(const AttributeTypes& attributeTypes)
  161. {
  162. Q_D(ctkVTKDataSetModel);
  163. if (d->AttributeType == attributeTypes)
  164. {
  165. return;
  166. }
  167. d->AttributeType = attributeTypes;
  168. this->updateDataSet();
  169. }
  170. //------------------------------------------------------------------------------
  171. vtkDataArray* ctkVTKDataSetModel::arrayFromItem(QStandardItem* dataArrayItem)const
  172. {
  173. if (dataArrayItem == 0 || dataArrayItem == this->invisibleRootItem())
  174. {
  175. return 0;
  176. }
  177. QVariant dataArrayPointer = dataArrayItem->data(ctkVTK::PointerRole);
  178. Q_ASSERT(dataArrayPointer.isValid());
  179. vtkDataArray* dataArray = static_cast<vtkDataArray*>(
  180. reinterpret_cast<void *>(dataArrayPointer.toLongLong()));
  181. Q_ASSERT(dataArray);
  182. return dataArray;
  183. }
  184. //------------------------------------------------------------------------------
  185. QStandardItem* ctkVTKDataSetModel::itemFromArray(vtkDataArray* dataArray, int column)const
  186. {
  187. if (dataArray == 0)
  188. {
  189. return 0;
  190. }
  191. QModelIndexList indexes = this->match(this->index(-1,-1), ctkVTK::PointerRole,
  192. reinterpret_cast<long long>(dataArray), 1,
  193. Qt::MatchExactly | Qt::MatchRecursive);
  194. while (indexes.size())
  195. {
  196. if (indexes[0].column() == column)
  197. {
  198. return this->itemFromIndex(indexes[0]);
  199. }
  200. indexes = this->match(indexes[0], ctkVTK::PointerRole,
  201. reinterpret_cast<long long>(dataArray), 1,
  202. Qt::MatchExactly | Qt::MatchRecursive);
  203. }
  204. return 0;
  205. }
  206. //------------------------------------------------------------------------------
  207. QModelIndexList ctkVTKDataSetModel::indexes(vtkDataArray* dataArray)const
  208. {
  209. return this->match(this->index(-1,-1), ctkVTK::PointerRole,
  210. QVariant::fromValue(reinterpret_cast<long long>(dataArray)),
  211. -1, Qt::MatchExactly | Qt::MatchRecursive);
  212. }
  213. /*
  214. //------------------------------------------------------------------------------
  215. void ctkVTKDataSetModel::setListenArrayModifiedEvent(bool listen)
  216. {
  217. Q_D(ctkVTKDataSetModel);
  218. if (d->ListenArrayModifiedEvent == listen)
  219. {
  220. return;
  221. }
  222. d->ListenArrayModifiedEvent = listen;
  223. d->listenArrayModifiedEvent();
  224. }
  225. //------------------------------------------------------------------------------
  226. bool ctkVTKDataSetModel::listenNodeModifiedEvent()const
  227. {
  228. Q_D(const ctkVTKDataSetModel);
  229. return d->ListenNodeModifiedEvent;
  230. }
  231. */
  232. //------------------------------------------------------------------------------
  233. void ctkVTKDataSetModel::updateDataSet()
  234. {
  235. Q_D(ctkVTKDataSetModel);
  236. this->setRowCount(0);
  237. if (d->DataSet.GetPointer() == 0)
  238. {
  239. return;
  240. }
  241. // Populate scene with nodes
  242. this->populateDataSet();
  243. }
  244. //------------------------------------------------------------------------------
  245. void ctkVTKDataSetModel::populateDataSet()
  246. {
  247. Q_D(ctkVTKDataSetModel);
  248. Q_ASSERT(d->DataSet);
  249. QList<vtkDataArray*> attributeArrays;
  250. attributeArrays << ctkVTKDataSetModelPrivate::attributeArrayToInsert(d->AttributeType, d->DataSet->GetPointData());
  251. attributeArrays << ctkVTKDataSetModelPrivate::attributeArrayToInsert(d->AttributeType, d->DataSet->GetCellData());
  252. foreach(vtkDataArray* attributeArray, attributeArrays)
  253. {
  254. this->insertArray(attributeArray);
  255. }
  256. }
  257. //------------------------------------------------------------------------------
  258. void ctkVTKDataSetModel::insertArray(vtkDataArray* dataArray)
  259. {
  260. this->insertArray(dataArray, this->rowCount());
  261. }
  262. //------------------------------------------------------------------------------
  263. void ctkVTKDataSetModel::insertArray(vtkDataArray* dataArray, int row)
  264. {
  265. Q_D(ctkVTKDataSetModel);
  266. Q_ASSERT(vtkDataArray::SafeDownCast(dataArray));
  267. QList<QStandardItem*> items;
  268. for (int i= 0; i < this->columnCount(); ++i)
  269. {
  270. QStandardItem* newArrayItem = new QStandardItem();
  271. this->updateItemFromArray(newArrayItem, dataArray, i);
  272. items.append(newArrayItem);
  273. }
  274. this->insertRow(row,items);
  275. // TODO: don't listen to nodes that are hidden from editors ?
  276. if (d->ListenDataArrayModifiedEvent)
  277. {
  278. qvtkConnect(dataArray, vtkCommand::ModifiedEvent,
  279. this, SLOT(onArrayModified(vtkObject*)));
  280. }
  281. }
  282. //------------------------------------------------------------------------------
  283. void ctkVTKDataSetModel::updateItemFromArray(QStandardItem* item, vtkDataArray* dataArray, int column)
  284. {
  285. item->setData(QVariant::fromValue(reinterpret_cast<long long>(dataArray)), ctkVTK::PointerRole);
  286. switch (column)
  287. {
  288. case 0:
  289. item->setText(QString(dataArray->GetName()));
  290. break;
  291. default:
  292. Q_ASSERT(column == 0);
  293. break;
  294. }
  295. }
  296. //------------------------------------------------------------------------------
  297. /*
  298. void ctkVTKDataSetModel::updateItemFromPointsArray(QStandardItem* item, vtkDataArray* dataArray, int column)
  299. {
  300. this->updateItemFromArray(item, dataArray, column);
  301. switch (column)
  302. {
  303. case 0:
  304. item->setIcon();
  305. break;
  306. default:
  307. Q_ASSERT(column == 0)
  308. break;
  309. }
  310. }
  311. //------------------------------------------------------------------------------
  312. void ctkVTKDataSetModel::updateItemFromCellsArray(QStandardItem* item, vtkDataArray* dataArray, int column)
  313. {
  314. this->updateItemFromArray(item, dataArray, column);
  315. switch (column)
  316. {
  317. case 0:
  318. item->setIcon();
  319. break;
  320. default:
  321. Q_ASSERT(column == 0)
  322. break;
  323. }
  324. }
  325. */
  326. //------------------------------------------------------------------------------
  327. void ctkVTKDataSetModel::updateArrayFromItem(vtkDataArray* dataArray, QStandardItem* item)
  328. {
  329. if (item->column() == 0)
  330. {
  331. dataArray->SetName(item->text().toLatin1());
  332. }
  333. }
  334. //------------------------------------------------------------------------------
  335. void ctkVTKDataSetModel::onDataSetModified(vtkObject* dataSet)
  336. {
  337. Q_UNUSED(dataSet);
  338. this->updateDataSet();
  339. }
  340. //------------------------------------------------------------------------------
  341. void ctkVTKDataSetModel::onArrayModified(vtkObject* array)
  342. {
  343. vtkDataArray* dataArray = vtkDataArray::SafeDownCast(array);
  344. Q_ASSERT(dataArray);
  345. QModelIndexList arrayIndexes = this->indexes(dataArray);
  346. foreach (QModelIndex index, arrayIndexes)
  347. {
  348. QStandardItem* item = this->itemFromIndex(index);
  349. this->updateItemFromArray(item, dataArray, item->column());
  350. }
  351. }
  352. //------------------------------------------------------------------------------
  353. void ctkVTKDataSetModel::onItemChanged(QStandardItem * item)
  354. {
  355. vtkDataArray* dataArray = this->arrayFromItem(item);
  356. Q_ASSERT(dataArray);
  357. this->updateArrayFromItem(dataArray, item);
  358. }