ctkVTKDataSetModel.cpp 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559
  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. vtkSmartPointer<vtkPointData> DataSetPointData;
  39. vtkSmartPointer<vtkCellData> DataSetCellData;
  40. bool ListenAbstractArrayModifiedEvent;
  41. ctkVTKDataSetModel::AttributeTypes AttributeType;
  42. bool IncludeNullItem;
  43. };
  44. //------------------------------------------------------------------------------
  45. ctkVTKDataSetModelPrivate::ctkVTKDataSetModelPrivate(ctkVTKDataSetModel& object)
  46. : q_ptr(&object)
  47. {
  48. this->ListenAbstractArrayModifiedEvent = false;
  49. this->AttributeType = ctkVTKDataSetModel::AllAttribute;
  50. this->IncludeNullItem = false;
  51. }
  52. //------------------------------------------------------------------------------
  53. ctkVTKDataSetModelPrivate::~ctkVTKDataSetModelPrivate()
  54. {
  55. }
  56. //------------------------------------------------------------------------------
  57. void ctkVTKDataSetModelPrivate::init()
  58. {
  59. Q_Q(ctkVTKDataSetModel);
  60. q->setColumnCount(1);
  61. QObject::connect(q, SIGNAL(itemChanged(QStandardItem*)),
  62. q, SLOT(onItemChanged(QStandardItem*)));
  63. }
  64. /*
  65. //------------------------------------------------------------------------------
  66. void ctkVTKDataSetModelPrivate::listenAbstractArrayModifiedEvent()
  67. {
  68. Q_Q(ctkVTKDataSetModel);
  69. q->qvtkDisconnect(0, vtkCommand::ModifiedEvent, q, SLOT(onArrayModified(vtkObject*)));
  70. if (!this->ListenAbstractArrayModifiedEvent)
  71. {
  72. return;
  73. }
  74. const int count = q->rowCount();
  75. for (int i = 0; i < count; ++i)
  76. {
  77. q->qvtkConnect(q->arrayFromIndex(q->index(i,0)),vtkCommand::ModifiedEvent,
  78. q, SLOT(onMRMLNodeModified(vtkObject*)));
  79. }
  80. }
  81. */
  82. //------------------------------------------------------------------------------
  83. QList<vtkAbstractArray*> ctkVTKDataSetModelPrivate::attributeArrayToInsert(
  84. const ctkVTKDataSetModel::AttributeTypes& attributeType,
  85. vtkDataSetAttributes * dataSetAttributes)
  86. {
  87. QList<vtkAbstractArray*> attributeArraysToInsert;
  88. for (int p = 0; p < dataSetAttributes->GetNumberOfArrays(); ++p)
  89. {
  90. vtkAbstractArray * array = dataSetAttributes->GetAbstractArray(p);
  91. bool isAttributeArray = false;
  92. vtkAbstractArray* attributeArrays[vtkDataSetAttributes::NUM_ATTRIBUTES];
  93. for(int attributeId = 0; attributeId < vtkDataSetAttributes::NUM_ATTRIBUTES; ++attributeId)
  94. {
  95. attributeArrays[attributeId] = dataSetAttributes->GetAbstractAttribute(attributeId);
  96. if (!isAttributeArray && attributeArrays[attributeId] == array)
  97. {
  98. isAttributeArray = true;
  99. }
  100. }
  101. if ((attributeType & ctkVTKDataSetModel::ScalarsAttribute && (array == attributeArrays[vtkDataSetAttributes::SCALARS]))
  102. || (attributeType & ctkVTKDataSetModel::VectorsAttribute && (array == attributeArrays[vtkDataSetAttributes::VECTORS]))
  103. || (attributeType & ctkVTKDataSetModel::NormalsAttribute && (array == attributeArrays[vtkDataSetAttributes::NORMALS]))
  104. || (attributeType & ctkVTKDataSetModel::TCoordsAttribute && (array == attributeArrays[vtkDataSetAttributes::TCOORDS]))
  105. || (attributeType & ctkVTKDataSetModel::TensorsAttribute && (array == attributeArrays[vtkDataSetAttributes::TENSORS]))
  106. || (attributeType & ctkVTKDataSetModel::GlobalIDsAttribute && (array == attributeArrays[vtkDataSetAttributes::GLOBALIDS]))
  107. || (attributeType & ctkVTKDataSetModel::PedigreeIDsAttribute && (array == attributeArrays[vtkDataSetAttributes::PEDIGREEIDS]))
  108. || (attributeType & ctkVTKDataSetModel::EdgeFlagAttribute && (array == attributeArrays[vtkDataSetAttributes::EDGEFLAG]))
  109. || (attributeType & ctkVTKDataSetModel::NoAttribute && !isAttributeArray)
  110. )
  111. {
  112. attributeArraysToInsert << dataSetAttributes->GetAbstractArray(p);
  113. }
  114. }
  115. return attributeArraysToInsert;
  116. }
  117. //------------------------------------------------------------------------------
  118. // ctkVTKDataSetModel
  119. //------------------------------------------------------------------------------
  120. ctkVTKDataSetModel::ctkVTKDataSetModel(QObject *_parent)
  121. : QStandardItemModel(_parent)
  122. , d_ptr(new ctkVTKDataSetModelPrivate(*this))
  123. , NullItemLocation(-2) // -1 is already used
  124. {
  125. Q_D(ctkVTKDataSetModel);
  126. d->init();
  127. }
  128. //------------------------------------------------------------------------------
  129. ctkVTKDataSetModel::ctkVTKDataSetModel(ctkVTKDataSetModelPrivate* pimpl, QObject *parentObject)
  130. : QStandardItemModel(parentObject)
  131. , d_ptr(pimpl)
  132. {
  133. Q_D(ctkVTKDataSetModel);
  134. d->init();
  135. }
  136. //------------------------------------------------------------------------------
  137. ctkVTKDataSetModel::~ctkVTKDataSetModel()
  138. {
  139. }
  140. //------------------------------------------------------------------------------
  141. void ctkVTKDataSetModel::setDataSet(vtkDataSet* dataSet)
  142. {
  143. Q_D(ctkVTKDataSetModel);
  144. if (dataSet == d->DataSet.GetPointer())
  145. {
  146. return;
  147. }
  148. this->qvtkReconnect(d->DataSet, dataSet, vtkCommand::ModifiedEvent,
  149. this, SLOT(onDataSetModified(vtkObject*)) );
  150. d->DataSet = dataSet;
  151. this->onDataSetModified(dataSet);
  152. }
  153. //------------------------------------------------------------------------------
  154. vtkDataSet* ctkVTKDataSetModel::dataSet()const
  155. {
  156. Q_D(const ctkVTKDataSetModel);
  157. return d->DataSet;
  158. }
  159. //------------------------------------------------------------------------------
  160. ctkVTKDataSetModel::AttributeTypes ctkVTKDataSetModel::attributeTypes()const
  161. {
  162. Q_D(const ctkVTKDataSetModel);
  163. return d->AttributeType;
  164. }
  165. //------------------------------------------------------------------------------
  166. void ctkVTKDataSetModel::setAttributeTypes(const AttributeTypes& attributeTypes)
  167. {
  168. Q_D(ctkVTKDataSetModel);
  169. if (d->AttributeType == attributeTypes)
  170. {
  171. return;
  172. }
  173. d->AttributeType = attributeTypes;
  174. this->updateDataSet();
  175. }
  176. // ----------------------------------------------------------------------------
  177. bool ctkVTKDataSetModel::includeNullItem()const
  178. {
  179. Q_D(const ctkVTKDataSetModel);
  180. return d->IncludeNullItem;
  181. }
  182. // ----------------------------------------------------------------------------
  183. void ctkVTKDataSetModel::setIncludeNullItem(bool includeNullItem)
  184. {
  185. Q_D(ctkVTKDataSetModel);
  186. if (d->IncludeNullItem == includeNullItem)
  187. {
  188. // no change
  189. return;
  190. }
  191. if (includeNullItem)
  192. {
  193. this->insertNullItem();
  194. }
  195. else
  196. {
  197. this->removeNullItem();
  198. }
  199. d->IncludeNullItem = includeNullItem;
  200. }
  201. //------------------------------------------------------------------------------
  202. vtkAbstractArray* ctkVTKDataSetModel::arrayFromItem(QStandardItem* arrayItem)const
  203. {
  204. if (arrayItem == 0 || arrayItem == this->invisibleRootItem())
  205. {
  206. return 0;
  207. }
  208. QVariant arrayPointer = arrayItem->data(ctkVTK::PointerRole);
  209. Q_ASSERT(arrayPointer.isValid());
  210. vtkAbstractArray* array = static_cast<vtkAbstractArray*>(
  211. reinterpret_cast<void *>(arrayPointer.toLongLong()));
  212. if (arrayItem->data(ctkVTK::LocationRole).toInt() == this->NullItemLocation)
  213. {
  214. // null item
  215. Q_ASSERT(array==0);
  216. return 0;
  217. }
  218. Q_ASSERT(array);
  219. return array;
  220. }
  221. //------------------------------------------------------------------------------
  222. int ctkVTKDataSetModel::locationFromItem(QStandardItem* arrayItem)const
  223. {
  224. if (arrayItem == 0 || arrayItem == this->invisibleRootItem())
  225. {
  226. return -1;
  227. }
  228. return arrayItem->data(ctkVTK::LocationRole).toInt();
  229. }
  230. //------------------------------------------------------------------------------
  231. QStandardItem* ctkVTKDataSetModel::itemFromArray(vtkAbstractArray* array, int column)const
  232. {
  233. if (array == 0)
  234. {
  235. return 0;
  236. }
  237. QModelIndexList indexes = this->match(this->index(-1,-1), ctkVTK::PointerRole,
  238. reinterpret_cast<long long>(array), 1,
  239. Qt::MatchExactly | Qt::MatchRecursive);
  240. while (indexes.size())
  241. {
  242. if (indexes[0].column() == column)
  243. {
  244. return this->itemFromIndex(indexes[0]);
  245. }
  246. indexes = this->match(indexes[0], ctkVTK::PointerRole,
  247. reinterpret_cast<long long>(array), 1,
  248. Qt::MatchExactly | Qt::MatchRecursive);
  249. }
  250. return 0;
  251. }
  252. //------------------------------------------------------------------------------
  253. QModelIndexList ctkVTKDataSetModel::indexes(vtkAbstractArray* array)const
  254. {
  255. return this->match(this->index(-1,-1), ctkVTK::PointerRole,
  256. QVariant::fromValue(reinterpret_cast<long long>(array)),
  257. -1, Qt::MatchExactly | Qt::MatchRecursive);
  258. }
  259. /*
  260. //------------------------------------------------------------------------------
  261. void ctkVTKDataSetModel::setListenArrayModifiedEvent(bool listen)
  262. {
  263. Q_D(ctkVTKDataSetModel);
  264. if (d->ListenArrayModifiedEvent == listen)
  265. {
  266. return;
  267. }
  268. d->ListenArrayModifiedEvent = listen;
  269. d->listenArrayModifiedEvent();
  270. }
  271. //------------------------------------------------------------------------------
  272. bool ctkVTKDataSetModel::listenNodeModifiedEvent()const
  273. {
  274. Q_D(const ctkVTKDataSetModel);
  275. return d->ListenNodeModifiedEvent;
  276. }
  277. */
  278. //------------------------------------------------------------------------------
  279. void ctkVTKDataSetModel::updateDataSet()
  280. {
  281. Q_D(ctkVTKDataSetModel);
  282. // Remove all items (except the first one, if there is a NULL item)
  283. if (d->IncludeNullItem)
  284. {
  285. if (this->rowCount()<1)
  286. {
  287. this->insertNullItem();
  288. }
  289. else
  290. {
  291. this->setRowCount(1);
  292. }
  293. }
  294. else
  295. {
  296. this->setRowCount(0);
  297. }
  298. if (d->DataSet.GetPointer() == 0)
  299. {
  300. return;
  301. }
  302. // Populate scene with nodes
  303. this->populateDataSet();
  304. }
  305. //------------------------------------------------------------------------------
  306. void ctkVTKDataSetModel::populateDataSet()
  307. {
  308. Q_D(ctkVTKDataSetModel);
  309. Q_ASSERT(d->DataSet);
  310. foreach(vtkAbstractArray* attributeArray,
  311. ctkVTKDataSetModelPrivate::attributeArrayToInsert(d->AttributeType, d->DataSet->GetPointData()))
  312. {
  313. this->insertArray(attributeArray, vtkAssignAttribute::POINT_DATA);
  314. }
  315. foreach(vtkAbstractArray* attributeArray,
  316. ctkVTKDataSetModelPrivate::attributeArrayToInsert(d->AttributeType, d->DataSet->GetCellData()))
  317. {
  318. this->insertArray(attributeArray, vtkAssignAttribute::CELL_DATA);
  319. }
  320. }
  321. //------------------------------------------------------------------------------
  322. void ctkVTKDataSetModel::insertArray(vtkAbstractArray* array, int location)
  323. {
  324. this->insertArray(array, location, this->rowCount());
  325. }
  326. //------------------------------------------------------------------------------
  327. void ctkVTKDataSetModel
  328. ::insertArray(vtkAbstractArray* array, int location, int row)
  329. {
  330. Q_D(ctkVTKDataSetModel);
  331. if (vtkAbstractArray::SafeDownCast(array)==0)
  332. {
  333. // it is normal, it happens when arrays are pre-allocated for a data set
  334. return;
  335. }
  336. QList<QStandardItem*> items;
  337. for (int i= 0; i < this->columnCount(); ++i)
  338. {
  339. QStandardItem* newArrayItem = new QStandardItem();
  340. this->updateItemFromArray(newArrayItem, array, location, i);
  341. items.append(newArrayItem);
  342. }
  343. this->insertRow(row,items);
  344. // TODO: don't listen to nodes that are hidden from editors ?
  345. if (d->ListenAbstractArrayModifiedEvent)
  346. {
  347. qvtkConnect(array, vtkCommand::ModifiedEvent,
  348. this, SLOT(onArrayModified(vtkObject*)));
  349. }
  350. }
  351. //------------------------------------------------------------------------------
  352. void ctkVTKDataSetModel::updateItemFromArray(QStandardItem* item,
  353. vtkAbstractArray* array,
  354. int location,
  355. int column)
  356. {
  357. item->setData(QVariant::fromValue(reinterpret_cast<long long>(array)), ctkVTK::PointerRole);
  358. item->setData(location, ctkVTK::LocationRole);
  359. switch (column)
  360. {
  361. case 0:
  362. item->setText(QString(array->GetName()));
  363. break;
  364. default:
  365. Q_ASSERT(column == 0);
  366. break;
  367. }
  368. }
  369. //------------------------------------------------------------------------------
  370. /*
  371. void ctkVTKDataSetModel::updateItemFromPointsArray(QStandardItem* item, vtkAbstractArray* array, int column)
  372. {
  373. this->updateItemFromArray(item, array, column);
  374. switch (column)
  375. {
  376. case 0:
  377. item->setIcon();
  378. break;
  379. default:
  380. Q_ASSERT(column == 0)
  381. break;
  382. }
  383. }
  384. //------------------------------------------------------------------------------
  385. void ctkVTKDataSetModel::updateItemFromCellsArray(QStandardItem* item, vtkAbstractArray* array, int column)
  386. {
  387. this->updateItemFromArray(item, array, column);
  388. switch (column)
  389. {
  390. case 0:
  391. item->setIcon();
  392. break;
  393. default:
  394. Q_ASSERT(column == 0)
  395. break;
  396. }
  397. }
  398. */
  399. //------------------------------------------------------------------------------
  400. void ctkVTKDataSetModel::updateArrayFromItem(vtkAbstractArray* array, QStandardItem* item)
  401. {
  402. if (item->column() == 0)
  403. {
  404. array->SetName(item->text().toLatin1());
  405. }
  406. }
  407. //------------------------------------------------------------------------------
  408. void ctkVTKDataSetModel::onDataSetModified(vtkObject* dataSet)
  409. {
  410. Q_UNUSED(dataSet);
  411. Q_D(ctkVTKDataSetModel);
  412. // If a point or cell data array is added or removed then DataSet's Modified is not invoked.
  413. // Therefore, we need to add observers to the point and cell data objects to make sure
  414. // the list of arrays is kept up-to-date.
  415. vtkPointData* dataSetPointData = d->DataSet ? d->DataSet->GetPointData() : 0;
  416. this->qvtkReconnect(d->DataSetPointData, dataSetPointData, vtkCommand::ModifiedEvent,
  417. this, SLOT(onDataSetPointDataModified(vtkObject*)) );
  418. d->DataSetPointData = dataSetPointData;
  419. vtkCellData* dataSetCellData = d->DataSet ? d->DataSet->GetCellData() : 0;
  420. this->qvtkReconnect(d->DataSetCellData, dataSetCellData, vtkCommand::ModifiedEvent,
  421. this, SLOT(onDataSetCellDataModified(vtkObject*)) );
  422. d->DataSetCellData = dataSetCellData;
  423. this->updateDataSet();
  424. }
  425. //------------------------------------------------------------------------------
  426. void ctkVTKDataSetModel::onDataSetPointDataModified(vtkObject* dataSetPointData)
  427. {
  428. Q_UNUSED(dataSetPointData);
  429. this->updateDataSet();
  430. }
  431. //------------------------------------------------------------------------------
  432. void ctkVTKDataSetModel::onDataSetCellDataModified(vtkObject* dataSetCellData)
  433. {
  434. Q_UNUSED(dataSetCellData);
  435. this->updateDataSet();
  436. }
  437. //------------------------------------------------------------------------------
  438. void ctkVTKDataSetModel::onArrayModified(vtkObject* modifiedArray)
  439. {
  440. vtkAbstractArray* array = vtkAbstractArray::SafeDownCast(modifiedArray);
  441. Q_ASSERT(array);
  442. QModelIndexList arrayIndexes = this->indexes(array);
  443. foreach (QModelIndex index, arrayIndexes)
  444. {
  445. QStandardItem* item = this->itemFromIndex(index);
  446. this->updateItemFromArray(
  447. item, array, item->data(ctkVTK::LocationRole).toInt(), item->column());
  448. }
  449. }
  450. //------------------------------------------------------------------------------
  451. void ctkVTKDataSetModel::onItemChanged(QStandardItem * item)
  452. {
  453. vtkAbstractArray* array = this->arrayFromItem(item);
  454. Q_ASSERT(array);
  455. this->updateArrayFromItem(array, item);
  456. }
  457. //------------------------------------------------------------------------------
  458. void ctkVTKDataSetModel::insertNullItem()
  459. {
  460. QStandardItem* nullItem = new QStandardItem();
  461. nullItem->setData(QVariant::fromValue(qlonglong(0)), ctkVTK::PointerRole);
  462. nullItem->setData(this->NullItemLocation, ctkVTK::LocationRole);
  463. nullItem->setText(QString());
  464. this->insertRow(0,nullItem);
  465. }
  466. //------------------------------------------------------------------------------
  467. void ctkVTKDataSetModel::removeNullItem()
  468. {
  469. if (this->rowCount() <= 0)
  470. {
  471. return;
  472. }
  473. // NULL item must be the first one
  474. QStandardItem* nullItem = this->item(0);
  475. Q_ASSERT(nullItem);
  476. if (nullItem == 0)
  477. {
  478. return;
  479. }
  480. // NULL item has a special location value
  481. int nullItemLocation = nullItem->data(ctkVTK::LocationRole).toInt();
  482. Q_ASSERT(nullItemLocation == this->NullItemLocation);
  483. if (nullItemLocation != this->NullItemLocation)
  484. {
  485. return;
  486. }
  487. // the first item is indeed the NULL item, so we remove it now
  488. this->removeRow(0);
  489. }
  490. //------------------------------------------------------------------------------
  491. int ctkVTKDataSetModel::nullItemLocation()const
  492. {
  493. return this->NullItemLocation;
  494. }