ctkVTKDataSetModel.cpp 14 KB

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