ctkDICOMAppWidget.cpp 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483
  1. // std includes
  2. #include <iostream>
  3. #include <dcmimage.h>
  4. // Qt includes
  5. #include <QDebug>
  6. #include <QTreeView>
  7. #include <QTabBar>
  8. #include <QSettings>
  9. #include <QAction>
  10. #include <QModelIndex>
  11. #include <QCheckBox>
  12. // ctkDICOMCore includes
  13. #include "ctkDICOMDatabase.h"
  14. #include "ctkDICOMIndexer.h"
  15. // ctkDICOMWidgets includes
  16. #include "ctkDICOMModel.h"
  17. #include "ctkDICOMAppWidget.h"
  18. #include "ctkDICOMQueryResultsTabWidget.h"
  19. #include "ui_ctkDICOMAppWidget.h"
  20. #include "ctkDirectoryButton.h"
  21. #include "ctkFileDialog.h"
  22. #include "ctkDICOMQueryRetrieveWidget.h"
  23. #include "ctkDICOMImportWidget.h"
  24. #include "ctkDICOMThumbnailWidget.h"
  25. #include "ctkDICOMThumbnailGenerator.h"
  26. //logger
  27. #include <ctkLogger.h>
  28. static ctkLogger logger("org.commontk.DICOM.Widgets.ctkDICOMAppWidget");
  29. //----------------------------------------------------------------------------
  30. class ctkDICOMAppWidgetPrivate: public Ui_ctkDICOMAppWidget
  31. {
  32. public:
  33. ctkDICOMAppWidgetPrivate();
  34. ctkFileDialog* ImportDialog;
  35. ctkDICOMQueryRetrieveWidget* QueryRetrieveWidget;
  36. QSharedPointer<ctkDICOMDatabase> DICOMDatabase;
  37. QSharedPointer<ctkDICOMThumbnailGenerator> thumbnailGenerator;
  38. ctkDICOMModel DICOMModel;
  39. QSharedPointer<ctkDICOMIndexer> DICOMIndexer;
  40. };
  41. //----------------------------------------------------------------------------
  42. // ctkDICOMAppWidgetPrivate methods
  43. ctkDICOMAppWidgetPrivate::ctkDICOMAppWidgetPrivate(){
  44. DICOMDatabase = QSharedPointer<ctkDICOMDatabase> (new ctkDICOMDatabase);
  45. thumbnailGenerator = QSharedPointer <ctkDICOMThumbnailGenerator> (new ctkDICOMThumbnailGenerator);
  46. DICOMDatabase->setThumbnailGenerator(thumbnailGenerator.data());
  47. DICOMIndexer = QSharedPointer<ctkDICOMIndexer> (new ctkDICOMIndexer);
  48. }
  49. //----------------------------------------------------------------------------
  50. // ctkDICOMAppWidget methods
  51. //----------------------------------------------------------------------------
  52. ctkDICOMAppWidget::ctkDICOMAppWidget(QWidget* _parent):Superclass(_parent),
  53. d_ptr(new ctkDICOMAppWidgetPrivate)
  54. {
  55. Q_D(ctkDICOMAppWidget);
  56. d->setupUi(this);
  57. //Hide image previewer buttons
  58. d->nextImageButton->hide();
  59. d->prevImageButton->hide();
  60. d->nextSeriesButton->hide();
  61. d->prevSeriesButton->hide();
  62. d->nextStudyButton->hide();
  63. d->prevStudyButton->hide();
  64. //Enable sorting in tree view
  65. d->treeView->setSortingEnabled(true);
  66. d->treeView->setSelectionBehavior(QAbstractItemView::SelectRows);
  67. connect(d->treeView, SIGNAL(collapsed(QModelIndex)), this, SLOT(onTreeCollapsed(QModelIndex)));
  68. connect(d->treeView, SIGNAL(expanded(QModelIndex)), this, SLOT(onTreeExpanded(QModelIndex)));
  69. //Set toolbar button style
  70. d->toolBar->setToolButtonStyle(Qt::ToolButtonTextUnderIcon);
  71. //Initialize Q/R widget
  72. d->QueryRetrieveWidget = new ctkDICOMQueryRetrieveWidget();
  73. d->QueryRetrieveWidget->setWindowModality ( Qt::ApplicationModal );
  74. //initialize directory from settings, then listen for changes
  75. QSettings settings;
  76. if ( settings.value("DatabaseDirectory", "") == "" )
  77. {
  78. QString directory = QString("./ctkDICOM-Database");
  79. settings.setValue("DatabaseDirectory", directory);
  80. settings.sync();
  81. }
  82. QString databaseDirectory = settings.value("DatabaseDirectory").toString();
  83. this->setDatabaseDirectory(databaseDirectory);
  84. d->directoryButton->setDirectory(databaseDirectory);
  85. connect(d->directoryButton, SIGNAL(directoryChanged(const QString&)), this, SLOT(setDatabaseDirectory(const QString&)));
  86. //Initialize import widget
  87. d->ImportDialog = new ctkFileDialog();
  88. QCheckBox* importCheckbox = new QCheckBox("Copy on import", d->ImportDialog);
  89. d->ImportDialog->setBottomWidget(importCheckbox);
  90. d->ImportDialog->setFileMode(QFileDialog::Directory);
  91. d->ImportDialog->setLabelText(QFileDialog::Accept,"Import");
  92. d->ImportDialog->setWindowTitle("Import DICOM files from directory ...");
  93. d->ImportDialog->setWindowModality(Qt::ApplicationModal);
  94. //connect signal and slots
  95. connect(d->treeView, SIGNAL(clicked(const QModelIndex&)), d->thumbnailsWidget, SLOT(onModelSelected(const QModelIndex &)));
  96. connect(d->treeView, SIGNAL(clicked(const QModelIndex&)), d->imagePreview, SLOT(onModelSelected(const QModelIndex &)));
  97. connect(d->treeView, SIGNAL(clicked(const QModelIndex&)), this, SLOT(onModelSelected(const QModelIndex &)));
  98. connect(d->thumbnailsWidget, SIGNAL(selected(const ctkDICOMThumbnailWidget&)), this, SLOT(onThumbnailSelected(const ctkDICOMThumbnailWidget&)));
  99. connect(d->thumbnailsWidget, SIGNAL(doubleClicked(const ctkDICOMThumbnailWidget&)), this, SLOT(onThumbnailDoubleClicked(const ctkDICOMThumbnailWidget&)));
  100. connect(d->ImportDialog, SIGNAL(fileSelected(QString)),this,SLOT(onImportDirectory(QString)));
  101. connect(d->DICOMDatabase.data(), SIGNAL( databaseChanged() ), &(d->DICOMModel), SLOT( reset() ) );
  102. connect(d->QueryRetrieveWidget, SIGNAL( canceled() ), d->QueryRetrieveWidget, SLOT( hide() ) );
  103. connect(d->imagePreview, SIGNAL(requestNextImage()), this, SLOT(onNextImage()));
  104. connect(d->imagePreview, SIGNAL(requestPreviousImage()), this, SLOT(onPreviousImage()));
  105. }
  106. //----------------------------------------------------------------------------
  107. ctkDICOMAppWidget::~ctkDICOMAppWidget()
  108. {
  109. Q_D(ctkDICOMAppWidget);
  110. d->QueryRetrieveWidget->deleteLater();
  111. d->ImportDialog->deleteLater();
  112. }
  113. //----------------------------------------------------------------------------
  114. void ctkDICOMAppWidget::setDatabaseDirectory(const QString& directory)
  115. {
  116. Q_D(ctkDICOMAppWidget);
  117. QSettings settings;
  118. settings.setValue("DatabaseDirectory", directory);
  119. settings.sync();
  120. //close the active DICOM database
  121. d->DICOMDatabase->closeDatabase();
  122. //open DICOM database on the directory
  123. QString databaseFileName = directory + QString("/ctkDICOM.sql");
  124. try { d->DICOMDatabase->openDatabase( databaseFileName ); }
  125. catch (std::exception e)
  126. {
  127. std::cerr << "Database error: " << qPrintable(d->DICOMDatabase->lastError()) << "\n";
  128. d->DICOMDatabase->closeDatabase();
  129. return;
  130. }
  131. d->DICOMModel.setDatabase(d->DICOMDatabase->database());
  132. d->DICOMModel.setDisplayLevel(ctkDICOMModel::SeriesType);
  133. d->treeView->setModel(&d->DICOMModel);
  134. d->treeView->resizeColumnToContents(0);
  135. //pass DICOM database instance to Import widget
  136. // d->ImportDialog->setDICOMDatabase(d->DICOMDatabase);
  137. d->QueryRetrieveWidget->setRetrieveDatabase(d->DICOMDatabase);
  138. // update the button and let any connected slots know about the change
  139. d->directoryButton->setDirectory(directory);
  140. d->thumbnailsWidget->setDatabaseDirectory(directory);
  141. d->imagePreview->setDatabaseDirectory(directory);
  142. emit databaseDirectoryChanged(directory);
  143. }
  144. //----------------------------------------------------------------------------
  145. QString ctkDICOMAppWidget::databaseDirectory() const
  146. {
  147. QSettings settings;
  148. return settings.value("DatabaseDirectory").toString();
  149. }
  150. void ctkDICOMAppWidget::onAddToDatabase()
  151. {
  152. //Q_D(ctkDICOMAppWidget);
  153. //d->
  154. }
  155. //----------------------------------------------------------------------------
  156. void ctkDICOMAppWidget::openImportDialog()
  157. {
  158. Q_D(ctkDICOMAppWidget);
  159. d->ImportDialog->show();
  160. d->ImportDialog->raise();
  161. }
  162. //----------------------------------------------------------------------------
  163. void ctkDICOMAppWidget::openExportDialog()
  164. {
  165. }
  166. //----------------------------------------------------------------------------
  167. void ctkDICOMAppWidget::openQueryDialog()
  168. {
  169. Q_D(ctkDICOMAppWidget);
  170. d->QueryRetrieveWidget->show();
  171. d->QueryRetrieveWidget->raise();
  172. }
  173. //----------------------------------------------------------------------------
  174. void ctkDICOMAppWidget::onThumbnailSelected(const ctkDICOMThumbnailWidget& widget)
  175. {
  176. Q_D(ctkDICOMAppWidget);
  177. d->imagePreview->onModelSelected(widget.sourceIndex());
  178. }
  179. //----------------------------------------------------------------------------
  180. void ctkDICOMAppWidget::onThumbnailDoubleClicked(const ctkDICOMThumbnailWidget& widget)
  181. {
  182. Q_D(ctkDICOMAppWidget);
  183. logger.debug("double clicked");
  184. QModelIndex index = widget.sourceIndex();
  185. ctkDICOMModel* model = const_cast<ctkDICOMModel*>(qobject_cast<const ctkDICOMModel*>(index.model()));
  186. QModelIndex index0 = index.sibling(index.row(), 0);
  187. if(model && (model->data(index0,ctkDICOMModel::TypeRole) != ctkDICOMModel::ImageType)){
  188. this->onModelSelected(index0);
  189. d->treeView->setCurrentIndex(index0);
  190. d->thumbnailsWidget->onModelSelected(index0);
  191. d->imagePreview->onModelSelected(index0);
  192. }
  193. }
  194. //----------------------------------------------------------------------------
  195. void ctkDICOMAppWidget::onImportDirectory(QString directory)
  196. {
  197. Q_D(ctkDICOMAppWidget);
  198. if (QDir(directory).exists())
  199. {
  200. QCheckBox* copyOnImport = qobject_cast<QCheckBox*>(d->ImportDialog->bottomWidget());
  201. QString targetDirectory;
  202. if (copyOnImport->isEnabled())
  203. {
  204. targetDirectory = d->DICOMDatabase->databaseDirectory();
  205. }
  206. d->DICOMIndexer->addDirectory(*d->DICOMDatabase,directory,targetDirectory);
  207. d->DICOMModel.reset();
  208. }
  209. }
  210. //----------------------------------------------------------------------------
  211. void ctkDICOMAppWidget::onModelSelected(const QModelIndex &index){
  212. Q_D(ctkDICOMAppWidget);
  213. ctkDICOMModel* model = const_cast<ctkDICOMModel*>(qobject_cast<const ctkDICOMModel*>(index.model()));
  214. if(model){
  215. QModelIndex index0 = index.sibling(index.row(), 0);
  216. if ( model->data(index0,ctkDICOMModel::TypeRole) == ctkDICOMModel::PatientType ){
  217. d->nextImageButton->show();
  218. d->prevImageButton->show();
  219. d->nextSeriesButton->show();
  220. d->prevSeriesButton->show();
  221. d->nextStudyButton->show();
  222. d->prevStudyButton->show();
  223. }else if ( model->data(index0,ctkDICOMModel::TypeRole) == ctkDICOMModel::StudyType ){
  224. d->nextImageButton->show();
  225. d->prevImageButton->show();
  226. d->nextSeriesButton->show();
  227. d->prevSeriesButton->show();
  228. d->nextStudyButton->hide();
  229. d->prevStudyButton->hide();
  230. }else if ( model->data(index0,ctkDICOMModel::TypeRole) == ctkDICOMModel::SeriesType ){
  231. d->nextImageButton->show();
  232. d->prevImageButton->show();
  233. d->nextSeriesButton->hide();
  234. d->prevSeriesButton->hide();
  235. d->nextStudyButton->hide();
  236. d->prevStudyButton->hide();
  237. }else{
  238. d->nextImageButton->hide();
  239. d->prevImageButton->hide();
  240. d->nextSeriesButton->hide();
  241. d->prevSeriesButton->hide();
  242. d->nextStudyButton->hide();
  243. d->prevStudyButton->hide();
  244. }
  245. }else{
  246. d->nextImageButton->hide();
  247. d->prevImageButton->hide();
  248. d->nextSeriesButton->hide();
  249. d->prevSeriesButton->hide();
  250. d->nextStudyButton->hide();
  251. d->prevStudyButton->hide();
  252. }
  253. }
  254. //----------------------------------------------------------------------------
  255. void ctkDICOMAppWidget::onNextImage(){
  256. Q_D(ctkDICOMAppWidget);
  257. QModelIndex currentIndex = d->imagePreview->currentImageIndex();
  258. if(currentIndex.isValid()){
  259. ctkDICOMModel* model = const_cast<ctkDICOMModel*>(qobject_cast<const ctkDICOMModel*>(currentIndex.model()));
  260. if(model){
  261. QModelIndex seriesIndex = currentIndex.parent();
  262. int imageCount = model->rowCount(seriesIndex);
  263. int imageID = currentIndex.row();
  264. imageID = (imageID+1)%imageCount;
  265. QModelIndex nextIndex = currentIndex.sibling(imageID, 0);
  266. d->imagePreview->onModelSelected(nextIndex);
  267. d->thumbnailsWidget->selectThumbnail(nextIndex);
  268. }
  269. }
  270. }
  271. //----------------------------------------------------------------------------
  272. void ctkDICOMAppWidget::onPreviousImage(){
  273. Q_D(ctkDICOMAppWidget);
  274. QModelIndex currentIndex = d->imagePreview->currentImageIndex();
  275. if(currentIndex.isValid()){
  276. ctkDICOMModel* model = const_cast<ctkDICOMModel*>(qobject_cast<const ctkDICOMModel*>(currentIndex.model()));
  277. if(model){
  278. QModelIndex seriesIndex = currentIndex.parent();
  279. int imageCount = model->rowCount(seriesIndex);
  280. int imageID = currentIndex.row();
  281. imageID--;
  282. if(imageID < 0) imageID += imageCount;
  283. QModelIndex prevIndex = currentIndex.sibling(imageID, 0);
  284. d->imagePreview->onModelSelected(prevIndex);
  285. d->thumbnailsWidget->selectThumbnail(prevIndex);
  286. }
  287. }
  288. }
  289. //----------------------------------------------------------------------------
  290. void ctkDICOMAppWidget::onNextSeries(){
  291. Q_D(ctkDICOMAppWidget);
  292. QModelIndex currentIndex = d->imagePreview->currentImageIndex();
  293. if(currentIndex.isValid()){
  294. ctkDICOMModel* model = const_cast<ctkDICOMModel*>(qobject_cast<const ctkDICOMModel*>(currentIndex.model()));
  295. if(model){
  296. QModelIndex seriesIndex = currentIndex.parent();
  297. QModelIndex studyIndex = seriesIndex.parent();
  298. int seriesCount = model->rowCount(studyIndex);
  299. int seriesID = seriesIndex.row();
  300. seriesID = (seriesID + 1)%seriesCount;
  301. QModelIndex nextIndex = seriesIndex.sibling(seriesID, 0);
  302. d->imagePreview->onModelSelected(nextIndex);
  303. d->thumbnailsWidget->selectThumbnail(nextIndex);
  304. }
  305. }
  306. }
  307. //----------------------------------------------------------------------------
  308. void ctkDICOMAppWidget::onPreviousSeries(){
  309. Q_D(ctkDICOMAppWidget);
  310. QModelIndex currentIndex = d->imagePreview->currentImageIndex();
  311. if(currentIndex.isValid()){
  312. ctkDICOMModel* model = const_cast<ctkDICOMModel*>(qobject_cast<const ctkDICOMModel*>(currentIndex.model()));
  313. if(model){
  314. QModelIndex seriesIndex = currentIndex.parent();
  315. QModelIndex studyIndex = seriesIndex.parent();
  316. int seriesCount = model->rowCount(studyIndex);
  317. int seriesID = seriesIndex.row();
  318. seriesID--;
  319. if(seriesID < 0) seriesID += seriesCount;
  320. QModelIndex prevIndex = seriesIndex.sibling(seriesID, 0);
  321. d->imagePreview->onModelSelected(prevIndex);
  322. d->thumbnailsWidget->selectThumbnail(prevIndex);
  323. }
  324. }
  325. }
  326. //----------------------------------------------------------------------------
  327. void ctkDICOMAppWidget::onNextStudy(){
  328. Q_D(ctkDICOMAppWidget);
  329. QModelIndex currentIndex = d->imagePreview->currentImageIndex();
  330. if(currentIndex.isValid()){
  331. ctkDICOMModel* model = const_cast<ctkDICOMModel*>(qobject_cast<const ctkDICOMModel*>(currentIndex.model()));
  332. if(model){
  333. QModelIndex seriesIndex = currentIndex.parent();
  334. QModelIndex studyIndex = seriesIndex.parent();
  335. QModelIndex patientIndex = studyIndex.parent();
  336. int studyCount = model->rowCount(patientIndex);
  337. int studyID = studyIndex.row();
  338. studyID = (studyID + 1)%studyCount;
  339. QModelIndex nextIndex = studyIndex.sibling(studyID, 0);
  340. d->imagePreview->onModelSelected(nextIndex);
  341. d->thumbnailsWidget->selectThumbnail(nextIndex);
  342. }
  343. }
  344. }
  345. //----------------------------------------------------------------------------
  346. void ctkDICOMAppWidget::onPreviousStudy(){
  347. Q_D(ctkDICOMAppWidget);
  348. QModelIndex currentIndex = d->imagePreview->currentImageIndex();
  349. if(currentIndex.isValid()){
  350. ctkDICOMModel* model = const_cast<ctkDICOMModel*>(qobject_cast<const ctkDICOMModel*>(currentIndex.model()));
  351. if(model){
  352. QModelIndex seriesIndex = currentIndex.parent();
  353. QModelIndex studyIndex = seriesIndex.parent();
  354. QModelIndex patientIndex = studyIndex.parent();
  355. int studyCount = model->rowCount(patientIndex);
  356. int studyID = studyIndex.row();
  357. studyID--;
  358. if(studyID < 0) studyID += studyCount;
  359. QModelIndex prevIndex = studyIndex.sibling(studyID, 0);
  360. d->imagePreview->onModelSelected(prevIndex);
  361. d->thumbnailsWidget->selectThumbnail(prevIndex);
  362. }
  363. }
  364. }
  365. //----------------------------------------------------------------------------
  366. void ctkDICOMAppWidget::onTreeCollapsed(const QModelIndex &index){
  367. Q_D(ctkDICOMAppWidget);
  368. d->treeView->resizeColumnToContents(0);
  369. }
  370. //----------------------------------------------------------------------------
  371. void ctkDICOMAppWidget::onTreeExpanded(const QModelIndex &index){
  372. Q_D(ctkDICOMAppWidget);
  373. d->treeView->resizeColumnToContents(0);
  374. }