ctkErrorLogModel.cpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498
  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.commontk.org/LICENSE
  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 <QApplication>
  16. #include <QDateTime>
  17. #include <QDebug>
  18. #include <QMainWindow>
  19. #include <QMetaEnum>
  20. #include <QMetaType>
  21. #include <QPointer>
  22. #include <QStandardItem>
  23. #include <QStatusBar>
  24. // CTK includes
  25. #include "ctkErrorLogModel.h"
  26. #include <ctkPimpl.h>
  27. // --------------------------------------------------------------------------
  28. // ctkErrorLogAbstractMessageHandler methods
  29. // --------------------------------------------------------------------------
  30. ctkErrorLogAbstractMessageHandler::ctkErrorLogAbstractMessageHandler()
  31. : Enabled(false)
  32. {
  33. }
  34. // --------------------------------------------------------------------------
  35. ctkErrorLogAbstractMessageHandler::~ctkErrorLogAbstractMessageHandler()
  36. {
  37. }
  38. // --------------------------------------------------------------------------
  39. ctkErrorLogModel* ctkErrorLogAbstractMessageHandler::errorLogModel()const
  40. {
  41. return this->ErrorLogModel.data();
  42. }
  43. // --------------------------------------------------------------------------
  44. void ctkErrorLogAbstractMessageHandler::setErrorLogModel(ctkErrorLogModel * newErrorLogModel)
  45. {
  46. this->ErrorLogModel = QPointer<ctkErrorLogModel>(newErrorLogModel);
  47. }
  48. // --------------------------------------------------------------------------
  49. QString ctkErrorLogAbstractMessageHandler::handlerPrettyName()const
  50. {
  51. if (this->HandlerPrettyName.isEmpty())
  52. {
  53. return this->handlerName();
  54. }
  55. else
  56. {
  57. return this->HandlerPrettyName;
  58. }
  59. }
  60. // --------------------------------------------------------------------------
  61. void ctkErrorLogAbstractMessageHandler::setHandlerPrettyName(const QString& newHandlerPrettyName)
  62. {
  63. this->HandlerPrettyName = newHandlerPrettyName;
  64. }
  65. // --------------------------------------------------------------------------
  66. bool ctkErrorLogAbstractMessageHandler::enabled()const
  67. {
  68. return this->Enabled;
  69. }
  70. // --------------------------------------------------------------------------
  71. void ctkErrorLogAbstractMessageHandler::setEnabled(bool value)
  72. {
  73. if (value == this->Enabled)
  74. {
  75. return;
  76. }
  77. this->setEnabledInternal(value);
  78. this->Enabled = value;
  79. }
  80. // --------------------------------------------------------------------------
  81. // ctkErrorLogModelPrivate
  82. // --------------------------------------------------------------------------
  83. class ctkErrorLogModelPrivate
  84. {
  85. Q_DECLARE_PUBLIC(ctkErrorLogModel);
  86. protected:
  87. ctkErrorLogModel* const q_ptr;
  88. public:
  89. ctkErrorLogModelPrivate(ctkErrorLogModel& object);
  90. ~ctkErrorLogModelPrivate();
  91. void init();
  92. QStandardItemModel StandardItemModel;
  93. QHash<QString, ctkErrorLogAbstractMessageHandler*> RegisteredHandlers;
  94. ctkErrorLogModel::LogLevels CurrentLogLevelFilter;
  95. bool LogEntryGrouping;
  96. int TerminalOutputEnabled;
  97. };
  98. // --------------------------------------------------------------------------
  99. // ctkErrorLogModelPrivate methods
  100. // --------------------------------------------------------------------------
  101. ctkErrorLogModelPrivate::ctkErrorLogModelPrivate(ctkErrorLogModel& object)
  102. : q_ptr(&object)
  103. {
  104. this->TerminalOutputEnabled = false;
  105. }
  106. // --------------------------------------------------------------------------
  107. ctkErrorLogModelPrivate::~ctkErrorLogModelPrivate()
  108. {
  109. this->LogEntryGrouping = false;
  110. foreach(const QString& handlerName, this->RegisteredHandlers.keys())
  111. {
  112. ctkErrorLogAbstractMessageHandler * msgHandler =
  113. this->RegisteredHandlers.value(handlerName);
  114. Q_ASSERT(msgHandler);
  115. msgHandler->setEnabled(false);
  116. delete msgHandler;
  117. }
  118. }
  119. // --------------------------------------------------------------------------
  120. void ctkErrorLogModelPrivate::init()
  121. {
  122. Q_Q(ctkErrorLogModel);
  123. q->setDynamicSortFilter(true);
  124. //
  125. // WARNING - Using a QSortFilterProxyModel slows down the insertion of rows by a factor 10
  126. //
  127. q->setSourceModel(&this->StandardItemModel);
  128. q->setFilterKeyColumn(ctkErrorLogModel::LogLevelColumn);
  129. }
  130. // --------------------------------------------------------------------------
  131. // ctkErrorLogModel methods
  132. //------------------------------------------------------------------------------
  133. ctkErrorLogModel::ctkErrorLogModel(QObject * parentObject)
  134. : Superclass(parentObject)
  135. , d_ptr(new ctkErrorLogModelPrivate(*this))
  136. {
  137. Q_D(ctkErrorLogModel);
  138. d->init();
  139. }
  140. //------------------------------------------------------------------------------
  141. ctkErrorLogModel::~ctkErrorLogModel()
  142. {
  143. }
  144. //------------------------------------------------------------------------------
  145. bool ctkErrorLogModel::registerMsgHandler(ctkErrorLogAbstractMessageHandler * msgHandler)
  146. {
  147. Q_D(ctkErrorLogModel);
  148. if (!msgHandler)
  149. {
  150. return false;
  151. }
  152. if (d->RegisteredHandlers.keys().contains(msgHandler->handlerName()))
  153. {
  154. return false;
  155. }
  156. msgHandler->setErrorLogModel(this);
  157. d->RegisteredHandlers.insert(msgHandler->handlerName(), msgHandler);
  158. return true;
  159. }
  160. //------------------------------------------------------------------------------
  161. QStringList ctkErrorLogModel::msgHandlerNames()const
  162. {
  163. Q_D(const ctkErrorLogModel);
  164. return d->RegisteredHandlers.keys();
  165. }
  166. //------------------------------------------------------------------------------
  167. bool ctkErrorLogModel::msgHandlerEnabled(const QString& handlerName) const
  168. {
  169. Q_D(const ctkErrorLogModel);
  170. if (!d->RegisteredHandlers.keys().contains(handlerName))
  171. {
  172. return false;
  173. }
  174. return d->RegisteredHandlers.value(handlerName)->enabled();
  175. }
  176. //------------------------------------------------------------------------------
  177. void ctkErrorLogModel::setMsgHandlerEnabled(const QString& handlerName, bool enabled)
  178. {
  179. Q_D(ctkErrorLogModel);
  180. if (!d->RegisteredHandlers.keys().contains(handlerName))
  181. {
  182. // qCritical() << "Failed to enable/disable message handler " << handlerName
  183. // << "- Handler not registered !";
  184. return;
  185. }
  186. d->RegisteredHandlers.value(handlerName)->setEnabled(enabled);
  187. }
  188. //------------------------------------------------------------------------------
  189. QStringList ctkErrorLogModel::msgHandlerEnabled() const
  190. {
  191. Q_D(const ctkErrorLogModel);
  192. QStringList msgHandlers;
  193. foreach(const QString& handlerName, d->RegisteredHandlers.keys())
  194. {
  195. if (d->RegisteredHandlers.value(handlerName)->enabled())
  196. {
  197. msgHandlers << handlerName;
  198. }
  199. }
  200. return msgHandlers;
  201. }
  202. //------------------------------------------------------------------------------
  203. void ctkErrorLogModel::setMsgHandlerEnabled(const QStringList& handlerNames)
  204. {
  205. foreach(const QString& handlerName, handlerNames)
  206. {
  207. this->setMsgHandlerEnabled(handlerName, true);
  208. }
  209. }
  210. //------------------------------------------------------------------------------
  211. void ctkErrorLogModel::enableAllMsgHandler()
  212. {
  213. this->setAllMsgHandlerEnabled(true);
  214. }
  215. //------------------------------------------------------------------------------
  216. void ctkErrorLogModel::disableAllMsgHandler()
  217. {
  218. this->setAllMsgHandlerEnabled(false);
  219. }
  220. //------------------------------------------------------------------------------
  221. void ctkErrorLogModel::setAllMsgHandlerEnabled(bool enabled)
  222. {
  223. Q_D(ctkErrorLogModel);
  224. foreach(const QString& msgHandlerName, d->RegisteredHandlers.keys())
  225. {
  226. this->setMsgHandlerEnabled(msgHandlerName, enabled);
  227. }
  228. }
  229. //------------------------------------------------------------------------------
  230. bool ctkErrorLogModel::terminalOutputEnabled()const
  231. {
  232. Q_D(const ctkErrorLogModel);
  233. return d->TerminalOutputEnabled;
  234. }
  235. //------------------------------------------------------------------------------
  236. void ctkErrorLogModel::setTerminalOutputEnabled(bool enabled)
  237. {
  238. Q_D(ctkErrorLogModel);
  239. d->TerminalOutputEnabled = enabled;
  240. }
  241. //------------------------------------------------------------------------------
  242. QString ctkErrorLogModel::logLevelAsString(LogLevel logLevel)const
  243. {
  244. QMetaEnum logLevelEnum = this->metaObject()->enumerator(0);
  245. Q_ASSERT(QString("LogLevel").compare(logLevelEnum.name()) == 0);
  246. return QLatin1String(logLevelEnum.valueToKey(logLevel));
  247. }
  248. //------------------------------------------------------------------------------
  249. void ctkErrorLogModel::addEntry(ctkErrorLogModel::LogLevel logLevel,
  250. const QString& origin, const char* text)
  251. {
  252. Q_D(ctkErrorLogModel);
  253. QString timeFormat("dd.MM.yyyy hh:mm:ss");
  254. QDateTime currentDateTime = QDateTime::currentDateTime();
  255. bool groupEntry = false;
  256. if (d->LogEntryGrouping)
  257. {
  258. // Retrieve logLevel associated with last row
  259. QModelIndex lastRowLogLevelIndex =
  260. d->StandardItemModel.index(d->StandardItemModel.rowCount() - 1, ctkErrorLogModel::LogLevelColumn);
  261. bool logLevelMatched = this->logLevelAsString(logLevel) == lastRowLogLevelIndex.data().toString();
  262. // Retrieve origin associated with last row
  263. QModelIndex lastRowOriginIndex =
  264. d->StandardItemModel.index(d->StandardItemModel.rowCount() - 1, ctkErrorLogModel::OriginColumn);
  265. bool originMatched = origin == lastRowOriginIndex.data().toString();
  266. // Retrieve time associated with last row
  267. QModelIndex lastRowTimeIndex =
  268. d->StandardItemModel.index(d->StandardItemModel.rowCount() - 1, ctkErrorLogModel::TimeColumn);
  269. QDateTime lastRowDateTime = QDateTime::fromString(lastRowTimeIndex.data().toString(), timeFormat);
  270. int groupingIntervalInMsecs = 1000;
  271. bool withinGroupingInterval = lastRowDateTime.time().msecsTo(currentDateTime.time()) <= groupingIntervalInMsecs;
  272. groupEntry = logLevelMatched && originMatched && withinGroupingInterval;
  273. }
  274. if (!groupEntry)
  275. {
  276. QList<QStandardItem*> itemList;
  277. // Time item
  278. QStandardItem * timeItem = new QStandardItem(currentDateTime.toString(timeFormat));
  279. timeItem->setEditable(false);
  280. itemList << timeItem;
  281. // LogLevel item
  282. QStandardItem * logLevelItem = new QStandardItem(this->logLevelAsString(logLevel));
  283. logLevelItem->setEditable(false);
  284. itemList << logLevelItem;
  285. // Origin item
  286. QStandardItem * originItem = new QStandardItem(origin);
  287. originItem->setEditable(false);
  288. itemList << originItem;
  289. // Description item
  290. QStandardItem * descriptionItem = new QStandardItem();
  291. QString descriptionText(text);
  292. descriptionItem->setData(descriptionText.left(160).append((descriptionText.size() > 160) ? "..." : ""), Qt::DisplayRole);
  293. descriptionItem->setData(descriptionText, ctkErrorLogModel::DescriptionTextRole);
  294. descriptionItem->setEditable(false);
  295. itemList << descriptionItem;
  296. d->StandardItemModel.invisibleRootItem()->appendRow(itemList);
  297. }
  298. else
  299. {
  300. // Retrieve description associated with last row
  301. QModelIndex lastRowDescriptionIndex =
  302. d->StandardItemModel.index(d->StandardItemModel.rowCount() - 1, ctkErrorLogModel::DescriptionColumn);
  303. QStringList updatedDescription;
  304. updatedDescription << lastRowDescriptionIndex.data(ctkErrorLogModel::DescriptionTextRole).toString();
  305. updatedDescription << QLatin1String(text);
  306. d->StandardItemModel.setData(lastRowDescriptionIndex, updatedDescription.join("\n"),
  307. ctkErrorLogModel::DescriptionTextRole);
  308. // Append '...' to displayText if needed
  309. QString displayText = lastRowDescriptionIndex.data().toString();
  310. if (!displayText.endsWith("..."))
  311. {
  312. d->StandardItemModel.setData(lastRowDescriptionIndex, displayText.append("..."), Qt::DisplayRole);
  313. }
  314. }
  315. if (d->TerminalOutputEnabled)
  316. {
  317. QStringList savedMsgHandlerEnabled = this->msgHandlerEnabled();
  318. this->disableAllMsgHandler();
  319. if (logLevel <= ctkErrorLogModel::Info)
  320. {
  321. std::cout << text << std::endl;
  322. }
  323. else
  324. {
  325. std::cerr << text << std::endl;
  326. }
  327. this->setMsgHandlerEnabled(savedMsgHandlerEnabled);
  328. }
  329. }
  330. //------------------------------------------------------------------------------
  331. void ctkErrorLogModel::clear()
  332. {
  333. Q_D(ctkErrorLogModel);
  334. d->StandardItemModel.invisibleRootItem()->removeRows(0, d->StandardItemModel.rowCount());
  335. }
  336. //------------------------------------------------------------------------------
  337. void ctkErrorLogModel::filterEntry(const LogLevels& logLevel, bool disableFilter)
  338. {
  339. Q_D(ctkErrorLogModel);
  340. QStringList patterns;
  341. if (!this->filterRegExp().pattern().isEmpty())
  342. {
  343. patterns << this->filterRegExp().pattern().split("|");
  344. }
  345. patterns.removeAll(this->logLevelAsString(Self::None));
  346. // foreach(QString s, patterns)
  347. // {
  348. // std::cout << "pattern:" << qPrintable(s) << std::endl;
  349. // }
  350. QMetaEnum logLevelEnum = this->metaObject()->enumerator(0);
  351. Q_ASSERT(QString("LogLevel").compare(logLevelEnum.name()) == 0);
  352. // Loop over enum values and append associated name to 'patterns' if
  353. // it has been specified within 'logLevel'
  354. for (int i = 1; i < logLevelEnum.keyCount(); ++i)
  355. {
  356. int aLogLevel = logLevelEnum.value(i);
  357. if (logLevel & aLogLevel)
  358. {
  359. QString logLevelAsString = this->logLevelAsString(static_cast<Self::LogLevel>(aLogLevel));
  360. if (!disableFilter)
  361. {
  362. patterns << logLevelAsString;
  363. d->CurrentLogLevelFilter |= static_cast<Self::LogLevels>(aLogLevel);
  364. }
  365. else
  366. {
  367. patterns.removeAll(logLevelAsString);
  368. d->CurrentLogLevelFilter ^= static_cast<Self::LogLevels>(aLogLevel);
  369. }
  370. }
  371. }
  372. if (patterns.isEmpty())
  373. {
  374. // If there are no patterns, let's filter with the None level so that
  375. // all entries are filtered out.
  376. patterns << this->logLevelAsString(Self::None);
  377. }
  378. bool filterChanged = true;
  379. QStringList currentPatterns = this->filterRegExp().pattern().split("|");
  380. if (currentPatterns.count() == patterns.count())
  381. {
  382. foreach(const QString& p, patterns)
  383. {
  384. currentPatterns.removeAll(p);
  385. }
  386. filterChanged = currentPatterns.count() > 0;
  387. }
  388. this->setFilterRegExp(patterns.join("|"));
  389. if (filterChanged)
  390. {
  391. emit this->logLevelFilterChanged();
  392. }
  393. }
  394. //------------------------------------------------------------------------------
  395. ctkErrorLogModel::LogLevels ctkErrorLogModel::logLevelFilter()const
  396. {
  397. QMetaEnum logLevelEnum = this->metaObject()->enumerator(0);
  398. Q_ASSERT(QString("LogLevel").compare(logLevelEnum.name()) == 0);
  399. Self::LogLevels filter = Self::Unknown;
  400. foreach(const QString& filterAsString, this->filterRegExp().pattern().split("|"))
  401. {
  402. filter |= static_cast<Self::LogLevels>(logLevelEnum.keyToValue(filterAsString.toLatin1()));
  403. }
  404. return filter;
  405. }
  406. //------------------------------------------------------------------------------
  407. bool ctkErrorLogModel::logEntryGrouping()const
  408. {
  409. Q_D(const ctkErrorLogModel);
  410. return d->LogEntryGrouping;
  411. }
  412. //------------------------------------------------------------------------------
  413. void ctkErrorLogModel::setLogEntryGrouping(bool value)
  414. {
  415. Q_D(ctkErrorLogModel);
  416. d->LogEntryGrouping = value;
  417. }