ctkDICOMServerNodeWidget.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352
  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. #include <QList>
  17. #include <QMap>
  18. #include <QSettings>
  19. #include <QTableWidgetItem>
  20. #include <QVariant>
  21. /// CTK includes
  22. #include <ctkCheckableHeaderView.h>
  23. #include <ctkCheckableModelHelper.h>
  24. // ctkDICOMWidgets includes
  25. #include "ctkDICOMServerNodeWidget.h"
  26. #include "ui_ctkDICOMServerNodeWidget.h"
  27. // STD includes
  28. #include <iostream>
  29. //----------------------------------------------------------------------------
  30. class ctkDICOMServerNodeWidgetPrivate: public Ui_ctkDICOMServerNodeWidget
  31. {
  32. public:
  33. ctkDICOMServerNodeWidgetPrivate(){}
  34. };
  35. //----------------------------------------------------------------------------
  36. // ctkDICOMServerNodeWidgetPrivate methods
  37. //----------------------------------------------------------------------------
  38. // ctkDICOMServerNodeWidget methods
  39. //----------------------------------------------------------------------------
  40. ctkDICOMServerNodeWidget::ctkDICOMServerNodeWidget(QWidget* parentWidget)
  41. : Superclass(parentWidget)
  42. , d_ptr(new ctkDICOMServerNodeWidgetPrivate)
  43. {
  44. Q_D(ctkDICOMServerNodeWidget);
  45. d->setupUi(this);
  46. // checkable headers.
  47. d->NodeTable->model()->setHeaderData(NameColumn, Qt::Horizontal, Qt::Unchecked, Qt::CheckStateRole);
  48. QHeaderView* previousHeaderView = d->NodeTable->horizontalHeader();
  49. ctkCheckableHeaderView* headerView = new ctkCheckableHeaderView(Qt::Horizontal, d->NodeTable);
  50. headerView->setClickable(previousHeaderView->isClickable());
  51. headerView->setMovable(previousHeaderView->isMovable());
  52. headerView->setHighlightSections(previousHeaderView->highlightSections());
  53. headerView->checkableModelHelper()->setPropagateDepth(-1);
  54. d->NodeTable->setHorizontalHeader(headerView);
  55. d->RemoveButton->setEnabled(false);
  56. this->readSettings();
  57. connect(d->CallingAETitle, SIGNAL(textChanged(QString)),
  58. this, SLOT(saveSettings()));
  59. connect(d->StorageAETitle, SIGNAL(textChanged(QString)),
  60. this, SLOT(saveSettings()));
  61. connect(d->StoragePort, SIGNAL(textChanged(QString)),
  62. this, SLOT(saveSettings()));
  63. connect(d->AddButton, SIGNAL(clicked()),
  64. this, SLOT(addServerNode()));
  65. connect(d->RemoveButton, SIGNAL(clicked()),
  66. this, SLOT(removeCurrentServerNode()));
  67. connect(d->NodeTable, SIGNAL(cellChanged(int,int)),
  68. this, SLOT(saveSettings()));
  69. connect(d->NodeTable, SIGNAL(currentItemChanged(QTableWidgetItem*,QTableWidgetItem*)),
  70. this, SLOT(updateRemoveButtonEnableState()));
  71. }
  72. //----------------------------------------------------------------------------
  73. ctkDICOMServerNodeWidget::~ctkDICOMServerNodeWidget()
  74. {
  75. }
  76. //----------------------------------------------------------------------------
  77. int ctkDICOMServerNodeWidget::addServerNode()
  78. {
  79. Q_D(ctkDICOMServerNodeWidget);
  80. const int rowCount = d->NodeTable->rowCount();
  81. d->NodeTable->setRowCount( rowCount + 1 );
  82. QTableWidgetItem* newItem = new QTableWidgetItem;
  83. newItem->setCheckState( Qt::Unchecked );
  84. d->NodeTable->setItem(rowCount, NameColumn, newItem);
  85. newItem = new QTableWidgetItem;
  86. newItem->setCheckState( Qt::Checked );
  87. d->NodeTable->setItem(rowCount, CGETColumn, newItem);
  88. d->NodeTable->setCurrentCell(rowCount, NameColumn);
  89. // The old rowCount becomes the added row index
  90. return rowCount;
  91. }
  92. //----------------------------------------------------------------------------
  93. int ctkDICOMServerNodeWidget::addServerNode(const QMap<QString, QVariant>& node)
  94. {
  95. Q_D(ctkDICOMServerNodeWidget);
  96. const int row = this->addServerNode();
  97. QTableWidgetItem *newItem;
  98. newItem = new QTableWidgetItem( node["Name"].toString() );
  99. newItem->setCheckState( Qt::CheckState(node["CheckState"].toInt()) );
  100. d->NodeTable->setItem(row, NameColumn, newItem);
  101. newItem = new QTableWidgetItem( node["AETitle"].toString() );
  102. d->NodeTable->setItem(row, AETitleColumn, newItem);
  103. newItem = new QTableWidgetItem( node["Address"].toString() );
  104. d->NodeTable->setItem(row, AddressColumn, newItem);
  105. newItem = new QTableWidgetItem( node["Port"].toString() );
  106. d->NodeTable->setItem(row, PortColumn, newItem);
  107. newItem = new QTableWidgetItem( QString("") );
  108. newItem->setCheckState( Qt::CheckState(node["CGET"].toInt()) );
  109. d->NodeTable->setItem(row, CGETColumn, newItem);
  110. return row;
  111. }
  112. //----------------------------------------------------------------------------
  113. void ctkDICOMServerNodeWidget::removeCurrentServerNode()
  114. {
  115. Q_D(ctkDICOMServerNodeWidget);
  116. d->NodeTable->removeRow( d->NodeTable->currentRow() );
  117. this->saveSettings();
  118. this->updateRemoveButtonEnableState();
  119. }
  120. //----------------------------------------------------------------------------
  121. void ctkDICOMServerNodeWidget::updateRemoveButtonEnableState()
  122. {
  123. Q_D(ctkDICOMServerNodeWidget);
  124. d->RemoveButton->setEnabled(d->NodeTable->rowCount() > 0);
  125. }
  126. //----------------------------------------------------------------------------
  127. void ctkDICOMServerNodeWidget::saveSettings()
  128. {
  129. Q_D(ctkDICOMServerNodeWidget);
  130. QSettings settings;
  131. const int rowCount = d->NodeTable->rowCount();
  132. settings.setValue("ServerNodeCount", rowCount);
  133. for (int row = 0; row < rowCount; ++row)
  134. {
  135. QMap<QString, QVariant> node = this->serverNodeParameters(row);
  136. settings.setValue(QString("ServerNodes/%1").arg(row), QVariant(node));
  137. }
  138. settings.setValue("CallingAETitle", this->callingAETitle());
  139. settings.setValue("StorageAETitle", this->storageAETitle());
  140. settings.setValue("StoragePort", this->storagePort());
  141. settings.sync();
  142. }
  143. //----------------------------------------------------------------------------
  144. void ctkDICOMServerNodeWidget::readSettings()
  145. {
  146. Q_D(ctkDICOMServerNodeWidget);
  147. d->NodeTable->setRowCount(0);
  148. QSettings settings;
  149. QMap<QString, QVariant> node;
  150. if (settings.status() == QSettings::AccessError ||
  151. settings.value("ServerNodeCount").toInt() == 0)
  152. {
  153. d->StorageAETitle->setText("CTKSTORE");
  154. d->StoragePort->setText("11112");
  155. d->CallingAETitle->setText("CTKSTORE");
  156. // a dummy example
  157. QMap<QString, QVariant> defaultServerNode;
  158. defaultServerNode["Name"] = QString("ExampleHost");
  159. defaultServerNode["CheckState"] = Qt::Unchecked;
  160. defaultServerNode["AETitle"] = QString("AETITLE");
  161. defaultServerNode["Address"] = QString("dicom.example.com");
  162. defaultServerNode["Port"] = QString("11112");
  163. defaultServerNode["CGET"] = Qt::Unchecked;
  164. this->addServerNode(defaultServerNode);
  165. // the uk example - see http://www.dicomserver.co.uk/
  166. // and http://www.medicalconnections.co.uk/
  167. defaultServerNode["Name"] = QString("MedicalConnections");
  168. defaultServerNode["CheckState"] = Qt::Unchecked;
  169. defaultServerNode["AETitle"] = QString("ANYAE");
  170. defaultServerNode["Address"] = QString("dicomserver.co.uk");
  171. defaultServerNode["Port"] = QString("11112");
  172. defaultServerNode["CGET"] = Qt::Checked;
  173. this->addServerNode(defaultServerNode);
  174. return;
  175. }
  176. d->StorageAETitle->setText(settings.value("StorageAETitle").toString());
  177. d->StoragePort->setText(settings.value("StoragePort").toString());
  178. d->CallingAETitle->setText(settings.value("CallingAETitle").toString());
  179. const int count = settings.value("ServerNodeCount").toInt();
  180. for (int row = 0; row < count; ++row)
  181. {
  182. node = settings.value(QString("ServerNodes/%1").arg(row)).toMap();
  183. this->addServerNode(node);
  184. }
  185. }
  186. //----------------------------------------------------------------------------
  187. QString ctkDICOMServerNodeWidget::callingAETitle()const
  188. {
  189. Q_D(const ctkDICOMServerNodeWidget);
  190. return d->CallingAETitle->text();
  191. }
  192. //----------------------------------------------------------------------------
  193. QString ctkDICOMServerNodeWidget::storageAETitle()const
  194. {
  195. Q_D(const ctkDICOMServerNodeWidget);
  196. return d->StorageAETitle->text();
  197. }
  198. //----------------------------------------------------------------------------
  199. int ctkDICOMServerNodeWidget::storagePort()const
  200. {
  201. Q_D(const ctkDICOMServerNodeWidget);
  202. bool ok = false;
  203. int port = d->StoragePort->text().toInt(&ok);
  204. Q_ASSERT(ok);
  205. return port;
  206. }
  207. //----------------------------------------------------------------------------
  208. QMap<QString,QVariant> ctkDICOMServerNodeWidget::parameters()const
  209. {
  210. QMap<QString, QVariant> parameters;
  211. parameters["CallingAETitle"] = this->callingAETitle();
  212. parameters["StorageAETitle"] = this->storageAETitle();
  213. parameters["StoragePort"] = this->storagePort();
  214. return parameters;
  215. }
  216. //----------------------------------------------------------------------------
  217. QStringList ctkDICOMServerNodeWidget::serverNodes()const
  218. {
  219. Q_D(const ctkDICOMServerNodeWidget);
  220. QStringList nodes;
  221. const int count = d->NodeTable->rowCount();
  222. for (int row = 0; row < count; ++row)
  223. {
  224. QTableWidgetItem* item = d->NodeTable->item(row,NameColumn);
  225. nodes << (item ? item->text() : QString(""));
  226. }
  227. // If there are duplicates, serverNodeParameters(QString) will behave
  228. // strangely
  229. Q_ASSERT(nodes.removeDuplicates() == 0);
  230. return nodes;
  231. }
  232. //----------------------------------------------------------------------------
  233. QStringList ctkDICOMServerNodeWidget::selectedServerNodes()const
  234. {
  235. Q_D(const ctkDICOMServerNodeWidget);
  236. QStringList nodes;
  237. const int count = d->NodeTable->rowCount();
  238. for (int row = 0; row < count; ++row)
  239. {
  240. QTableWidgetItem* item = d->NodeTable->item(row, NameColumn);
  241. if (item && item->checkState() == Qt::Checked)
  242. {
  243. nodes << item->text();
  244. }
  245. }
  246. // If there are duplicates, serverNodeParameters(QString) will behave
  247. // strangely
  248. Q_ASSERT(nodes.removeDuplicates() == 0);
  249. return nodes;
  250. }
  251. //----------------------------------------------------------------------------
  252. QMap<QString, QVariant> ctkDICOMServerNodeWidget::serverNodeParameters(const QString &node)const
  253. {
  254. Q_D(const ctkDICOMServerNodeWidget);
  255. QMap<QString, QVariant> parameters;
  256. const int count = d->NodeTable->rowCount();
  257. for (int row = 0; row < count; row++)
  258. {
  259. if ( d->NodeTable->item(row,0)->text() == node )
  260. {
  261. // TBD: not sure what it means to merge parameters
  262. parameters.unite(this->serverNodeParameters(row));
  263. }
  264. }
  265. return parameters;
  266. }
  267. //----------------------------------------------------------------------------
  268. QMap<QString, QVariant> ctkDICOMServerNodeWidget::serverNodeParameters(int row)const
  269. {
  270. Q_D(const ctkDICOMServerNodeWidget);
  271. QMap<QString, QVariant> node;
  272. if (row < 0 || row >= d->NodeTable->rowCount())
  273. {
  274. return node;
  275. }
  276. const int columnCount = d->NodeTable->columnCount();
  277. for (int column = 0; column < columnCount; ++column)
  278. {
  279. if (!d->NodeTable->item(row, column))
  280. {
  281. continue;
  282. }
  283. QString label = d->NodeTable->horizontalHeaderItem(column)->text();
  284. node[label] = d->NodeTable->item(row, column)->data(Qt::DisplayRole);
  285. }
  286. node["CheckState"] = d->NodeTable->item(row, NameColumn) ?
  287. d->NodeTable->item(row,NameColumn)->checkState() : Qt::Unchecked;
  288. node["CGET"] = d->NodeTable->item(row, CGETColumn) ?
  289. d->NodeTable->item(row,CGETColumn)->checkState() : Qt::Unchecked;
  290. return node;
  291. }