qCTKDCMTKModel.cxx 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434
  1. #include <QStringList>
  2. #include <QSqlDriver>
  3. #include <QSqlError>
  4. #include <QSqlQuery>
  5. #include <QSqlQueryModel>
  6. #include <QSqlRecord>
  7. #include <QTime>
  8. #include <QDebug>
  9. #include "qCTKDCMTKModel.h"
  10. struct Node;
  11. class qCTKDCMTKModelPrivate:public qCTKPrivate<qCTKDCMTKModel>
  12. {
  13. public:
  14. qCTKDCMTKModelPrivate();
  15. ~qCTKDCMTKModelPrivate();
  16. void init();
  17. enum IndexType{
  18. RootType,
  19. PatientType,
  20. StudyType,
  21. SeriesType,
  22. ImageType
  23. };
  24. void fetch(const QModelIndex& index, int limit);
  25. Node* createNode(int row, int column, const QModelIndex& parent)const;
  26. Node* nodeFromIndex(const QModelIndex& index)const;
  27. QVariant value(const QModelIndex& index, int row, int field)const;
  28. QString generateQuery(const QString& fields, const QString& table, const QString& conditions = QString())const;
  29. void updateQueries(Node* node)const;
  30. Node* RootNode;
  31. QSqlDatabase DataBase;
  32. QStringList Headers;
  33. QString Sort;
  34. };
  35. struct Node
  36. {
  37. ~Node()
  38. {
  39. foreach(Node* node, this->Children)
  40. {
  41. delete node;
  42. }
  43. this->Children.clear();
  44. }
  45. qCTKDCMTKModelPrivate::IndexType Type;
  46. Node* Parent;
  47. QVector<Node*> Children;
  48. int Row;
  49. int Column;
  50. QSqlQuery Query;
  51. QString UID;
  52. int RowCount;
  53. bool End;
  54. bool Fetching;
  55. };
  56. qCTKDCMTKModelPrivate::qCTKDCMTKModelPrivate()
  57. {
  58. this->RootNode = 0;
  59. }
  60. qCTKDCMTKModelPrivate::~qCTKDCMTKModelPrivate()
  61. {
  62. delete this->RootNode;
  63. this->RootNode = 0;
  64. }
  65. void qCTKDCMTKModelPrivate::init()
  66. {
  67. QCTK_P(qCTKDCMTKModel);
  68. this->Headers = QStringList() << "Name" << "Age" << "Scan" << "Date" << "Subject ID"
  69. << "Number" << "Institution" << "Referrer" << "Performer";
  70. }
  71. Node* qCTKDCMTKModelPrivate::nodeFromIndex(const QModelIndex& index)const
  72. {
  73. return index.isValid() ? reinterpret_cast<Node*>(index.internalPointer()) : this->RootNode;
  74. }
  75. Node* qCTKDCMTKModelPrivate::createNode(int row, int column, const QModelIndex& parent)const
  76. {
  77. QCTK_P(const qCTKDCMTKModel);
  78. Node* node = new Node;
  79. Node* nodeParent = 0;
  80. if (row == -1 || column == -1)
  81. {// root node
  82. node->Type = qCTKDCMTKModelPrivate::RootType;
  83. node->Parent = 0;
  84. }
  85. else
  86. {
  87. nodeParent = this->nodeFromIndex(parent);
  88. nodeParent->Children.push_back(node);
  89. node->Parent = (nodeParent == this->RootNode ? 0: nodeParent);
  90. node->Type = qCTKDCMTKModelPrivate::IndexType(nodeParent->Type + 1);
  91. }
  92. node->Row = row;
  93. node->Column = column;
  94. if (node->Type != qCTKDCMTKModelPrivate::RootType)
  95. {
  96. int field = nodeParent->Query.record().indexOf("UID");
  97. node->UID = this->value(parent, row, field).toString();
  98. }
  99. node->RowCount = 0;
  100. node->End = false;
  101. node->Fetching = false;
  102. this->updateQueries(node);
  103. return node;
  104. }
  105. QVariant qCTKDCMTKModelPrivate::value(const QModelIndex& parent, int row, int column) const
  106. {
  107. Node* node = this->nodeFromIndex(parent);
  108. if (row >= node->RowCount)
  109. {
  110. const_cast<qCTKDCMTKModelPrivate *>(this)->fetch(parent, row + 256);
  111. }
  112. if (!node->Query.seek(row))
  113. {
  114. qDebug() << node->Query.lastError();
  115. return QVariant();
  116. }
  117. return node->Query.value(column);
  118. }
  119. QString qCTKDCMTKModelPrivate::generateQuery(const QString& fields, const QString& table, const QString& conditions)const
  120. {
  121. QString res = QString("SELECT ") + fields + QString(" FROM ") + table;
  122. if (!conditions.isEmpty())
  123. {
  124. res += QString(" WHERE ") + conditions;
  125. }
  126. if (this->Sort.isEmpty())
  127. {
  128. res += QString(" ORDER BY ") + this->Sort;
  129. }
  130. return res;
  131. }
  132. void qCTKDCMTKModelPrivate::updateQueries(Node* node)const
  133. {
  134. // are you kidding me, it should be virtualized here :-)
  135. QString query;
  136. switch(node->Type)
  137. {
  138. default:
  139. Q_ASSERT(node->Type == qCTKDCMTKModelPrivate::RootType);
  140. break;
  141. case qCTKDCMTKModelPrivate::RootType:
  142. //query = QString("SELECT FROM ");
  143. query = this->generateQuery("UID as UID, PatientsName as Name, PatientsAge as Age, PatientsBirthDate as Date, PatientID as 'Subject ID'","Patients");
  144. break;
  145. case qCTKDCMTKModelPrivate::PatientType:
  146. //query = QString("SELECT FROM Studies WHERE PatientsUID='%1'").arg(node->UID);
  147. query = this->generateQuery("StudyInstanceUID as UID, StudyDescription as Name, ModalitiesInStudy as Scan, StudyDate as Date, AccessionNumber as Number, ReferringPhysician as Institution, ReferringPhysician as Referrer, PerformingPysiciansName as Performer", "Studies",QString("PatientsUID='%1'").arg(node->UID));
  148. break;
  149. case qCTKDCMTKModelPrivate::StudyType:
  150. //query = QString("SELECT SeriesInstanceUID as UID, SeriesDescription as Name, BodyPartExamined as Scan, SeriesDate as Date, AcquisitionNumber as Number FROM Series WHERE StudyInstanceUID='%1'").arg(node->UID);
  151. query = this->generateQuery("SeriesInstanceUID as UID, SeriesDescription as Name, BodyPartExamined as Scan, SeriesDate as Date, AcquisitionNumber as Number","Series",QString("StudyInstanceUID='%1'").arg(node->UID));
  152. break;
  153. case qCTKDCMTKModelPrivate::SeriesType:
  154. //query = QString("SELECT Filename as UID, Filename as Name, SeriesInstanceUID as Date FROM Images WHERE SeriesInstanceUID='%1'").arg(node->UID);
  155. query = this->generateQuery("Filename as UID, Filename as Name, SeriesInstanceUID as Date", "Images", QString("SeriesInstanceUID='%1'").arg(node->UID));
  156. break;
  157. case qCTKDCMTKModelPrivate::ImageType:
  158. break;
  159. }
  160. node->Query = QSqlQuery(query, this->DataBase);
  161. foreach(Node* child, node->Children)
  162. {
  163. this->updateQueries(child);
  164. }
  165. }
  166. void qCTKDCMTKModelPrivate::fetch(const QModelIndex& index, int limit)
  167. {
  168. QCTK_P(qCTKDCMTKModel);
  169. Node* node = this->nodeFromIndex(index);
  170. if (node->End || limit <= node->RowCount || node->Fetching/*|| bottom.column() == -1*/)
  171. {
  172. return;
  173. }
  174. node->Fetching = true;
  175. int newRowCount;
  176. const int oldRowCount = node->RowCount;
  177. // try to seek directly
  178. if (node->Query.seek(limit - 1))
  179. {
  180. newRowCount = limit;
  181. }
  182. else
  183. {
  184. newRowCount = qMax(oldRowCount, 1);
  185. if (node->Query.seek(newRowCount - 1))
  186. {
  187. while (node->Query.next())
  188. {
  189. ++newRowCount;
  190. }
  191. }
  192. else
  193. {
  194. // empty or invalid query
  195. newRowCount = 0;
  196. }
  197. node->End = true; // this is the end.
  198. }
  199. if (newRowCount > 0 && newRowCount > node->RowCount)
  200. {
  201. p->beginInsertRows(index, node->RowCount, newRowCount - 1);
  202. node->RowCount = newRowCount;
  203. node->Fetching = false;
  204. p->endInsertRows();
  205. }
  206. else
  207. {
  208. node->RowCount = newRowCount;
  209. node->Fetching = false;
  210. }
  211. }
  212. qCTKDCMTKModel::qCTKDCMTKModel(QObject* parent)
  213. {
  214. QCTK_INIT_PRIVATE(qCTKDCMTKModel);
  215. qctk_d()->init();
  216. }
  217. qCTKDCMTKModel::~qCTKDCMTKModel()
  218. {
  219. }
  220. bool qCTKDCMTKModel::canFetchMore ( const QModelIndex & parent ) const
  221. {
  222. QCTK_D(const qCTKDCMTKModel);
  223. Node* node = d->nodeFromIndex(parent);
  224. return !node->End;
  225. }
  226. int qCTKDCMTKModel::columnCount ( const QModelIndex & _parent ) const
  227. {
  228. QCTK_D(const qCTKDCMTKModel);
  229. Q_UNUSED(_parent);
  230. return d->Headers.size();
  231. }
  232. QVariant qCTKDCMTKModel::data ( const QModelIndex & index, int role ) const
  233. {
  234. QCTK_D(const qCTKDCMTKModel);
  235. if (role & ~(Qt::DisplayRole | Qt::EditRole))
  236. {
  237. return QVariant();
  238. }
  239. QModelIndex indexParent = this->parent(index);
  240. Node* node = d->nodeFromIndex(indexParent);
  241. Q_ASSERT(node->Row == indexParent.row());
  242. if (index.row() >= node->RowCount)
  243. {
  244. const_cast<qCTKDCMTKModelPrivate *>(d)->fetch(index, index.row());
  245. }
  246. if (!node->Query.seek(index.row()))
  247. {
  248. qDebug() << node->Query.lastError();
  249. return QVariant();
  250. }
  251. int field = node->Query.record().indexOf(d->Headers[index.column()]);
  252. if (field < 0)
  253. {
  254. return QVariant();
  255. }
  256. return d->value(indexParent, index.row(), field);
  257. //return node->Query.value(field);
  258. }
  259. void qCTKDCMTKModel::fetchMore ( const QModelIndex & parent )
  260. {
  261. QCTK_D(qCTKDCMTKModel);
  262. Node* node = d->nodeFromIndex(parent);
  263. d->fetch(parent, qMax(node->RowCount, 0) + 256);
  264. }
  265. Qt::ItemFlags qCTKDCMTKModel::flags ( const QModelIndex & index ) const
  266. {
  267. return Qt::ItemIsSelectable | Qt::ItemIsEnabled;
  268. }
  269. bool qCTKDCMTKModel::hasChildren ( const QModelIndex & parent ) const
  270. {
  271. QCTK_D(const qCTKDCMTKModel);
  272. Node* node = d->nodeFromIndex(parent);
  273. return node->RowCount > 0 || (!node->End && node->Query.seek(0));
  274. }
  275. QVariant qCTKDCMTKModel::headerData(int section, Qt::Orientation orientation, int role)const
  276. {
  277. QCTK_D(const qCTKDCMTKModel);
  278. if (role & ~(Qt::DisplayRole | Qt::EditRole))
  279. {
  280. return QVariant();
  281. }
  282. if (orientation == Qt::Vertical)
  283. {
  284. return section;
  285. }
  286. Q_ASSERT(orientation == Qt::Horizontal);
  287. Q_ASSERT(section < d->Headers.size());
  288. return d->Headers[section];
  289. }
  290. QModelIndex qCTKDCMTKModel::index ( int row, int column, const QModelIndex & parent ) const
  291. {
  292. QCTK_D(const qCTKDCMTKModel);
  293. Node* parentNode = d->nodeFromIndex(parent);
  294. Node* node = 0;
  295. foreach(Node* tmpNode, parentNode->Children)
  296. {
  297. if (tmpNode->Row == row &&
  298. tmpNode->Column == column)
  299. {
  300. node = tmpNode;
  301. break;
  302. }
  303. }
  304. if (node == 0)
  305. {
  306. node = d->createNode(row, column, parent);
  307. }
  308. return this->createIndex(row, column, node);
  309. }
  310. QModelIndex qCTKDCMTKModel::parent ( const QModelIndex & index ) const
  311. {
  312. QCTK_D(const qCTKDCMTKModel);
  313. Node* node = d->nodeFromIndex(index);
  314. if (node == 0 || node->Parent == 0)
  315. {
  316. return QModelIndex();
  317. }
  318. return this->createIndex(node->Parent->Row, node->Parent->Column, node->Parent);
  319. }
  320. int qCTKDCMTKModel::rowCount ( const QModelIndex & parent ) const
  321. {
  322. QCTK_D(const qCTKDCMTKModel);
  323. Node* node = d->nodeFromIndex(parent);
  324. if (node->RowCount == 0 && node->End)
  325. {
  326. const_cast<qCTKDCMTKModelPrivate*>(d)->fetch(parent, 256);
  327. }
  328. return node->RowCount;
  329. }
  330. void qCTKDCMTKModel::setDatabase(const QSqlDatabase &db)
  331. {
  332. QCTK_D(qCTKDCMTKModel);
  333. this->beginResetModel();
  334. d->DataBase = db;
  335. delete d->RootNode;
  336. d->RootNode = 0;
  337. if (d->DataBase.tables().empty())
  338. {
  339. Q_ASSERT(d->DataBase.isOpen());
  340. return;
  341. }
  342. d->RootNode = d->createNode(-1, -1, QModelIndex());
  343. this->endResetModel();
  344. bool hasQuerySize = d->RootNode->Query.driver()->hasFeature(QSqlDriver::QuerySize);
  345. if (hasQuerySize && d->RootNode->Query.size() > 0)
  346. {
  347. int newRowCount= d->RootNode->Query.size();
  348. beginInsertRows(QModelIndex(), 0, qMax(0, newRowCount - 1));
  349. d->RootNode->RowCount = newRowCount;
  350. d->RootNode->End = true;
  351. endInsertRows();
  352. }
  353. else
  354. {
  355. d->RootNode->RowCount = 0;
  356. }
  357. d->fetch(QModelIndex(), 256);
  358. }
  359. void qCTKDCMTKModel::sort(int column, Qt::SortOrder order)
  360. {
  361. QCTK_D(qCTKDCMTKModel);
  362. emit layoutAboutToBeChanged();
  363. d->Sort = QString("'%1' %2")
  364. .arg(d->Headers[column])
  365. .arg(order == Qt::AscendingOrder ? "ASC" : "DESC");
  366. d->updateQueries(d->RootNode);
  367. emit layoutChanged();
  368. }
  369. bool qCTKDCMTKModel::setHeaderData ( int section, Qt::Orientation orientation, const QVariant & value, int role)
  370. {
  371. QCTK_D(qCTKDCMTKModel);
  372. if (role & ~(Qt::DisplayRole | Qt::EditRole))
  373. {
  374. return false;
  375. }
  376. if (orientation == Qt::Vertical)
  377. {
  378. return false;
  379. }
  380. if (value.toString() == d->Headers[section])
  381. {
  382. return false;
  383. }
  384. d->Headers[section] = value.toString();
  385. emit this->headerDataChanged(orientation, section, section);
  386. return true;
  387. }