ctkErrorLogModel.cpp 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776
  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 <QApplication>
  16. #include <QDateTime>
  17. #include <QDebug>
  18. #include <QFile>
  19. #include <QMainWindow>
  20. #include <QMetaEnum>
  21. #include <QMetaType>
  22. #include <QMutexLocker>
  23. #include <QPointer>
  24. #include <QStandardItem>
  25. #include <QStatusBar>
  26. // CTK includes
  27. #include "ctkErrorLogModel.h"
  28. #include <ctkPimpl.h>
  29. // STD includes
  30. #include <cstdio> // For _fileno or fileno
  31. #ifdef _MSC_VER
  32. # include <io.h> // For _write()
  33. #endif
  34. // --------------------------------------------------------------------------
  35. // ctkErrorLogLevel methods
  36. // --------------------------------------------------------------------------
  37. ctkErrorLogLevel::ctkErrorLogLevel()
  38. {
  39. qRegisterMetaType<ctkErrorLogLevel::LogLevel>("ctkErrorLogLevel::LogLevel");
  40. }
  41. // --------------------------------------------------------------------------
  42. QString ctkErrorLogLevel::operator()(ctkErrorLogLevel::LogLevel logLevel)
  43. {
  44. return this->logLevelAsString(logLevel);
  45. }
  46. // --------------------------------------------------------------------------
  47. QString ctkErrorLogLevel::logLevelAsString(ctkErrorLogLevel::LogLevel logLevel)const
  48. {
  49. QMetaEnum logLevelEnum = this->metaObject()->enumerator(0);
  50. Q_ASSERT(QString("LogLevel").compare(logLevelEnum.name()) == 0);
  51. return QLatin1String(logLevelEnum.valueToKey(logLevel));
  52. }
  53. // --------------------------------------------------------------------------
  54. // ctkErrorLogTerminalOutputPrivate
  55. // --------------------------------------------------------------------------
  56. class ctkErrorLogTerminalOutputPrivate
  57. {
  58. public:
  59. ctkErrorLogTerminalOutputPrivate();
  60. ~ctkErrorLogTerminalOutputPrivate();
  61. bool Enabled;
  62. mutable QMutex EnableMutex;
  63. int FD;
  64. mutable QMutex OutputMutex;
  65. };
  66. // --------------------------------------------------------------------------
  67. ctkErrorLogTerminalOutputPrivate::ctkErrorLogTerminalOutputPrivate()
  68. : Enabled(false)
  69. {
  70. #ifdef Q_OS_WIN32
  71. this->FD = _fileno(stdout);
  72. #else
  73. this->FD = fileno(stdout);
  74. #endif
  75. }
  76. // --------------------------------------------------------------------------
  77. ctkErrorLogTerminalOutputPrivate::~ctkErrorLogTerminalOutputPrivate()
  78. {
  79. }
  80. // --------------------------------------------------------------------------
  81. // ctkErrorLogTerminalOutput methods
  82. // --------------------------------------------------------------------------
  83. ctkErrorLogTerminalOutput::ctkErrorLogTerminalOutput()
  84. : d_ptr(new ctkErrorLogTerminalOutputPrivate)
  85. {
  86. }
  87. // --------------------------------------------------------------------------
  88. ctkErrorLogTerminalOutput::~ctkErrorLogTerminalOutput()
  89. {
  90. }
  91. // --------------------------------------------------------------------------
  92. bool ctkErrorLogTerminalOutput::enabled()const
  93. {
  94. Q_D(const ctkErrorLogTerminalOutput);
  95. QMutexLocker locker(&d->EnableMutex);
  96. return d->Enabled;
  97. }
  98. // --------------------------------------------------------------------------
  99. void ctkErrorLogTerminalOutput::setEnabled(bool value)
  100. {
  101. Q_D(ctkErrorLogTerminalOutput);
  102. QMutexLocker locker(&d->EnableMutex);
  103. d->Enabled = value;
  104. }
  105. // --------------------------------------------------------------------------
  106. int ctkErrorLogTerminalOutput::fileDescriptor()const
  107. {
  108. Q_D(const ctkErrorLogTerminalOutput);
  109. QMutexLocker locker(&d->OutputMutex);
  110. return d->FD;
  111. }
  112. // --------------------------------------------------------------------------
  113. void ctkErrorLogTerminalOutput::setFileDescriptor(int fd)
  114. {
  115. Q_D(ctkErrorLogTerminalOutput);
  116. QMutexLocker locker(&d->OutputMutex);
  117. d->FD = fd;
  118. }
  119. // --------------------------------------------------------------------------
  120. void ctkErrorLogTerminalOutput::output(const QString& text)
  121. {
  122. Q_D(ctkErrorLogTerminalOutput);
  123. {
  124. QMutexLocker locker(&d->EnableMutex);
  125. if (!d->Enabled)
  126. {
  127. return;
  128. }
  129. }
  130. {
  131. QMutexLocker locker(&d->OutputMutex);
  132. QString textWithNewLine = text + "\n";
  133. #ifdef _MSC_VER
  134. _write(d->FD, qPrintable(textWithNewLine), textWithNewLine.size());
  135. #else
  136. write(d->FD, qPrintable(textWithNewLine), textWithNewLine.size());
  137. #endif
  138. }
  139. }
  140. // --------------------------------------------------------------------------
  141. // ctkErrorLogModelPrivate
  142. // --------------------------------------------------------------------------
  143. class ctkErrorLogModelPrivate
  144. {
  145. Q_DECLARE_PUBLIC(ctkErrorLogModel);
  146. protected:
  147. ctkErrorLogModel* const q_ptr;
  148. public:
  149. ctkErrorLogModelPrivate(ctkErrorLogModel& object);
  150. ~ctkErrorLogModelPrivate();
  151. void init();
  152. /// Convenient method that could be used for debugging purposes.
  153. void appendToFile(const QString& fileName, const QString& text);
  154. void setMessageHandlerConnection(ctkErrorLogAbstractMessageHandler * msgHandler, bool asynchronous);
  155. QStandardItemModel StandardItemModel;
  156. QHash<QString, ctkErrorLogAbstractMessageHandler*> RegisteredHandlers;
  157. ctkErrorLogLevel::LogLevels CurrentLogLevelFilter;
  158. bool LogEntryGrouping;
  159. bool AsynchronousLogging;
  160. bool AddingEntry;
  161. ctkErrorLogLevel ErrorLogLevel;
  162. ctkErrorLogTerminalOutput StdErrTerminalOutput;
  163. ctkErrorLogTerminalOutput StdOutTerminalOutput;
  164. QMutex AppendToFileMutex;
  165. };
  166. // --------------------------------------------------------------------------
  167. // ctkErrorLogModelPrivate methods
  168. // --------------------------------------------------------------------------
  169. ctkErrorLogModelPrivate::ctkErrorLogModelPrivate(ctkErrorLogModel& object)
  170. : q_ptr(&object)
  171. {
  172. this->LogEntryGrouping = false;
  173. this->AsynchronousLogging = true;
  174. this->AddingEntry = false;
  175. }
  176. // --------------------------------------------------------------------------
  177. ctkErrorLogModelPrivate::~ctkErrorLogModelPrivate()
  178. {
  179. foreach(const QString& handlerName, this->RegisteredHandlers.keys())
  180. {
  181. ctkErrorLogAbstractMessageHandler * msgHandler =
  182. this->RegisteredHandlers.value(handlerName);
  183. Q_ASSERT(msgHandler);
  184. msgHandler->setEnabled(false);
  185. delete msgHandler;
  186. }
  187. }
  188. // --------------------------------------------------------------------------
  189. void ctkErrorLogModelPrivate::init()
  190. {
  191. Q_Q(ctkErrorLogModel);
  192. q->setDynamicSortFilter(true);
  193. //
  194. // WARNING - Using a QSortFilterProxyModel slows down the insertion of rows by a factor 10
  195. //
  196. q->setSourceModel(&this->StandardItemModel);
  197. q->setFilterKeyColumn(ctkErrorLogModel::LogLevelColumn);
  198. }
  199. // --------------------------------------------------------------------------
  200. void ctkErrorLogModelPrivate::appendToFile(const QString& fileName, const QString& text)
  201. {
  202. QMutexLocker locker(&this->AppendToFileMutex);
  203. QFile f(fileName);
  204. f.open(QFile::Append);
  205. QTextStream s(&f);
  206. s << QDateTime::currentDateTime().toString() << " - " << text << "\n";
  207. f.close();
  208. }
  209. // --------------------------------------------------------------------------
  210. void ctkErrorLogModelPrivate::setMessageHandlerConnection(
  211. ctkErrorLogAbstractMessageHandler * msgHandler, bool asynchronous)
  212. {
  213. Q_Q(ctkErrorLogModel);
  214. msgHandler->disconnect();
  215. QObject::connect(msgHandler,
  216. SIGNAL(messageHandled(QDateTime,QString,ctkErrorLogLevel::LogLevel,QString,QString)),
  217. q, SLOT(addEntry(QDateTime,QString,ctkErrorLogLevel::LogLevel,QString,QString)),
  218. asynchronous ? Qt::QueuedConnection : Qt::BlockingQueuedConnection);
  219. }
  220. // --------------------------------------------------------------------------
  221. // ctkErrorLogModel methods
  222. //------------------------------------------------------------------------------
  223. ctkErrorLogModel::ctkErrorLogModel(QObject * parentObject)
  224. : Superclass(parentObject)
  225. , d_ptr(new ctkErrorLogModelPrivate(*this))
  226. {
  227. Q_D(ctkErrorLogModel);
  228. d->init();
  229. }
  230. //------------------------------------------------------------------------------
  231. ctkErrorLogModel::~ctkErrorLogModel()
  232. {
  233. }
  234. //------------------------------------------------------------------------------
  235. bool ctkErrorLogModel::registerMsgHandler(ctkErrorLogAbstractMessageHandler * msgHandler)
  236. {
  237. Q_D(ctkErrorLogModel);
  238. if (!msgHandler)
  239. {
  240. return false;
  241. }
  242. if (d->RegisteredHandlers.keys().contains(msgHandler->handlerName()))
  243. {
  244. return false;
  245. }
  246. d->setMessageHandlerConnection(msgHandler, d->AsynchronousLogging);
  247. msgHandler->setTerminalOutput(Self::StandardError, &d->StdErrTerminalOutput);
  248. msgHandler->setTerminalOutput(Self::StandardOutput, &d->StdOutTerminalOutput);
  249. d->RegisteredHandlers.insert(msgHandler->handlerName(), msgHandler);
  250. return true;
  251. }
  252. //------------------------------------------------------------------------------
  253. QStringList ctkErrorLogModel::msgHandlerNames()const
  254. {
  255. Q_D(const ctkErrorLogModel);
  256. return d->RegisteredHandlers.keys();
  257. }
  258. //------------------------------------------------------------------------------
  259. bool ctkErrorLogModel::msgHandlerEnabled(const QString& handlerName) const
  260. {
  261. Q_D(const ctkErrorLogModel);
  262. if (!d->RegisteredHandlers.keys().contains(handlerName))
  263. {
  264. return false;
  265. }
  266. return d->RegisteredHandlers.value(handlerName)->enabled();
  267. }
  268. //------------------------------------------------------------------------------
  269. void ctkErrorLogModel::setMsgHandlerEnabled(const QString& handlerName, bool enabled)
  270. {
  271. Q_D(ctkErrorLogModel);
  272. if (!d->RegisteredHandlers.keys().contains(handlerName))
  273. {
  274. // qCritical() << "Failed to enable/disable message handler " << handlerName
  275. // << "- Handler not registered !";
  276. return;
  277. }
  278. d->RegisteredHandlers.value(handlerName)->setEnabled(enabled);
  279. }
  280. //------------------------------------------------------------------------------
  281. QStringList ctkErrorLogModel::msgHandlerEnabled() const
  282. {
  283. Q_D(const ctkErrorLogModel);
  284. QStringList msgHandlers;
  285. foreach(const QString& handlerName, d->RegisteredHandlers.keys())
  286. {
  287. if (d->RegisteredHandlers.value(handlerName)->enabled())
  288. {
  289. msgHandlers << handlerName;
  290. }
  291. }
  292. return msgHandlers;
  293. }
  294. //------------------------------------------------------------------------------
  295. void ctkErrorLogModel::setMsgHandlerEnabled(const QStringList& handlerNames)
  296. {
  297. foreach(const QString& handlerName, handlerNames)
  298. {
  299. this->setMsgHandlerEnabled(handlerName, true);
  300. }
  301. }
  302. //------------------------------------------------------------------------------
  303. void ctkErrorLogModel::enableAllMsgHandler()
  304. {
  305. this->setAllMsgHandlerEnabled(true);
  306. }
  307. //------------------------------------------------------------------------------
  308. void ctkErrorLogModel::disableAllMsgHandler()
  309. {
  310. this->setAllMsgHandlerEnabled(false);
  311. }
  312. //------------------------------------------------------------------------------
  313. void ctkErrorLogModel::setAllMsgHandlerEnabled(bool enabled)
  314. {
  315. Q_D(ctkErrorLogModel);
  316. foreach(const QString& msgHandlerName, d->RegisteredHandlers.keys())
  317. {
  318. this->setMsgHandlerEnabled(msgHandlerName, enabled);
  319. }
  320. }
  321. //------------------------------------------------------------------------------
  322. ctkErrorLogModel::TerminalOutputs ctkErrorLogModel::terminalOutputs()const
  323. {
  324. Q_D(const ctkErrorLogModel);
  325. ctkErrorLogModel::TerminalOutputs currentTerminalOutputs;
  326. currentTerminalOutputs |= d->StdErrTerminalOutput.enabled() ? Self::StandardError : Self::None;
  327. currentTerminalOutputs |= d->StdOutTerminalOutput.enabled() ? Self::StandardOutput : Self::None;
  328. return currentTerminalOutputs;
  329. }
  330. //------------------------------------------------------------------------------
  331. void ctkErrorLogModel::setTerminalOutputs(
  332. const ctkErrorLogModel::TerminalOutputs& terminalOutput)
  333. {
  334. Q_D(ctkErrorLogModel);
  335. d->StdErrTerminalOutput.setEnabled(terminalOutput & ctkErrorLogModel::StandardOutput);
  336. d->StdOutTerminalOutput.setEnabled(terminalOutput & ctkErrorLogModel::StandardError);
  337. }
  338. //------------------------------------------------------------------------------
  339. void ctkErrorLogModel::addEntry(const QDateTime& currentDateTime, const QString& threadId,
  340. ctkErrorLogLevel::LogLevel logLevel,
  341. const QString& origin, const QString& text)
  342. {
  343. Q_D(ctkErrorLogModel);
  344. // d->appendToFile("/tmp/ctkErrorLogModel-appendToFile.txt",
  345. // QString("addEntry: %1").arg(QThread::currentThreadId()));
  346. if (d->AddingEntry)
  347. {
  348. // QString str;
  349. // QTextStream s(&str);
  350. // s << "----------------------------------\n";
  351. // s << "text=>" << text << "\n";
  352. // s << "\tlogLevel:" << qPrintable(d->ErrorLogLevel(logLevel)) << "\n";
  353. // s << "\torigin:" << qPrintable(origin) << "\n";
  354. // d->appendToFile("/tmp/ctkErrorLogModel-AddingEntry-true.txt", str);
  355. return;
  356. }
  357. d->AddingEntry = true;
  358. QString timeFormat("dd.MM.yyyy hh:mm:ss");
  359. bool groupEntry = false;
  360. if (d->LogEntryGrouping)
  361. {
  362. // Retrieve threadId associated with last row
  363. QModelIndex lastRowThreadIdIndex =
  364. d->StandardItemModel.index(d->StandardItemModel.rowCount() - 1, ctkErrorLogModel::ThreadIdColumn);
  365. bool threadIdMatched = threadId == lastRowThreadIdIndex.data().toString();
  366. // Retrieve logLevel associated with last row
  367. QModelIndex lastRowLogLevelIndex =
  368. d->StandardItemModel.index(d->StandardItemModel.rowCount() - 1, ctkErrorLogModel::LogLevelColumn);
  369. bool logLevelMatched = d->ErrorLogLevel(logLevel) == lastRowLogLevelIndex.data().toString();
  370. // Retrieve origin associated with last row
  371. QModelIndex lastRowOriginIndex =
  372. d->StandardItemModel.index(d->StandardItemModel.rowCount() - 1, ctkErrorLogModel::OriginColumn);
  373. bool originMatched = origin == lastRowOriginIndex.data().toString();
  374. // Retrieve time associated with last row
  375. QModelIndex lastRowTimeIndex =
  376. d->StandardItemModel.index(d->StandardItemModel.rowCount() - 1, ctkErrorLogModel::TimeColumn);
  377. QDateTime lastRowDateTime = QDateTime::fromString(lastRowTimeIndex.data().toString(), timeFormat);
  378. int groupingIntervalInMsecs = 1000;
  379. bool withinGroupingInterval = lastRowDateTime.time().msecsTo(currentDateTime.time()) <= groupingIntervalInMsecs;
  380. groupEntry = threadIdMatched && logLevelMatched && originMatched && withinGroupingInterval;
  381. }
  382. if (!groupEntry)
  383. {
  384. QList<QStandardItem*> itemList;
  385. // Time item
  386. QStandardItem * timeItem = new QStandardItem(currentDateTime.toString(timeFormat));
  387. timeItem->setEditable(false);
  388. itemList << timeItem;
  389. // ThreadId item
  390. QStandardItem * threadIdItem = new QStandardItem(threadId);
  391. threadIdItem->setEditable(false);
  392. itemList << threadIdItem;
  393. // LogLevel item
  394. QStandardItem * logLevelItem = new QStandardItem(d->ErrorLogLevel(logLevel));
  395. logLevelItem->setEditable(false);
  396. itemList << logLevelItem;
  397. // Origin item
  398. QStandardItem * originItem = new QStandardItem(origin);
  399. originItem->setEditable(false);
  400. itemList << originItem;
  401. // Description item
  402. QStandardItem * descriptionItem = new QStandardItem();
  403. QString descriptionText(text);
  404. descriptionItem->setData(descriptionText.left(160).append((descriptionText.size() > 160) ? "..." : ""), Qt::DisplayRole);
  405. descriptionItem->setData(descriptionText, ctkErrorLogModel::DescriptionTextRole);
  406. descriptionItem->setEditable(false);
  407. itemList << descriptionItem;
  408. d->StandardItemModel.invisibleRootItem()->appendRow(itemList);
  409. }
  410. else
  411. {
  412. // Retrieve description associated with last row
  413. QModelIndex lastRowDescriptionIndex =
  414. d->StandardItemModel.index(d->StandardItemModel.rowCount() - 1, ctkErrorLogModel::DescriptionColumn);
  415. QStringList updatedDescription;
  416. updatedDescription << lastRowDescriptionIndex.data(ctkErrorLogModel::DescriptionTextRole).toString();
  417. updatedDescription << text;
  418. d->StandardItemModel.setData(lastRowDescriptionIndex, updatedDescription.join("\n"),
  419. ctkErrorLogModel::DescriptionTextRole);
  420. // Append '...' to displayText if needed
  421. QString displayText = lastRowDescriptionIndex.data().toString();
  422. if (!displayText.endsWith("..."))
  423. {
  424. d->StandardItemModel.setData(lastRowDescriptionIndex, displayText.append("..."), Qt::DisplayRole);
  425. }
  426. }
  427. d->AddingEntry = false;
  428. }
  429. //------------------------------------------------------------------------------
  430. void ctkErrorLogModel::clear()
  431. {
  432. Q_D(ctkErrorLogModel);
  433. d->StandardItemModel.invisibleRootItem()->removeRows(0, d->StandardItemModel.rowCount());
  434. }
  435. //------------------------------------------------------------------------------
  436. void ctkErrorLogModel::filterEntry(const ctkErrorLogLevel::LogLevels& logLevel,
  437. bool disableFilter)
  438. {
  439. Q_D(ctkErrorLogModel);
  440. QStringList patterns;
  441. if (!this->filterRegExp().pattern().isEmpty())
  442. {
  443. patterns << this->filterRegExp().pattern().split("|");
  444. }
  445. patterns.removeAll(d->ErrorLogLevel(ctkErrorLogLevel::None));
  446. // foreach(QString s, patterns)
  447. // {
  448. // std::cout << "pattern:" << qPrintable(s) << std::endl;
  449. // }
  450. QMetaEnum logLevelEnum = d->ErrorLogLevel.metaObject()->enumerator(0);
  451. Q_ASSERT(QString("LogLevel").compare(logLevelEnum.name()) == 0);
  452. // Loop over enum values and append associated name to 'patterns' if
  453. // it has been specified within 'logLevel'
  454. for (int i = 1; i < logLevelEnum.keyCount(); ++i)
  455. {
  456. int aLogLevel = logLevelEnum.value(i);
  457. if (logLevel & aLogLevel)
  458. {
  459. QString logLevelAsString = d->ErrorLogLevel(static_cast<ctkErrorLogLevel::LogLevel>(aLogLevel));
  460. if (!disableFilter)
  461. {
  462. patterns << logLevelAsString;
  463. d->CurrentLogLevelFilter |= static_cast<ctkErrorLogLevel::LogLevels>(aLogLevel);
  464. }
  465. else
  466. {
  467. patterns.removeAll(logLevelAsString);
  468. d->CurrentLogLevelFilter ^= static_cast<ctkErrorLogLevel::LogLevels>(aLogLevel);
  469. }
  470. }
  471. }
  472. if (patterns.isEmpty())
  473. {
  474. // If there are no patterns, let's filter with the None level so that
  475. // all entries are filtered out.
  476. patterns << d->ErrorLogLevel(ctkErrorLogLevel::None);
  477. }
  478. bool filterChanged = true;
  479. QStringList currentPatterns = this->filterRegExp().pattern().split("|");
  480. if (currentPatterns.count() == patterns.count())
  481. {
  482. foreach(const QString& p, patterns)
  483. {
  484. currentPatterns.removeAll(p);
  485. }
  486. filterChanged = currentPatterns.count() > 0;
  487. }
  488. this->setFilterRegExp(patterns.join("|"));
  489. if (filterChanged)
  490. {
  491. emit this->logLevelFilterChanged();
  492. }
  493. }
  494. //------------------------------------------------------------------------------
  495. ctkErrorLogLevel::LogLevels ctkErrorLogModel::logLevelFilter()const
  496. {
  497. Q_D(const ctkErrorLogModel);
  498. QMetaEnum logLevelEnum = d->ErrorLogLevel.metaObject()->enumerator(0);
  499. Q_ASSERT(QString("LogLevel").compare(logLevelEnum.name()) == 0);
  500. ctkErrorLogLevel::LogLevels filter = ctkErrorLogLevel::Unknown;
  501. foreach(const QString& filterAsString, this->filterRegExp().pattern().split("|"))
  502. {
  503. filter |= static_cast<ctkErrorLogLevel::LogLevels>(logLevelEnum.keyToValue(filterAsString.toLatin1()));
  504. }
  505. return filter;
  506. }
  507. //------------------------------------------------------------------------------
  508. bool ctkErrorLogModel::logEntryGrouping()const
  509. {
  510. Q_D(const ctkErrorLogModel);
  511. return d->LogEntryGrouping;
  512. }
  513. //------------------------------------------------------------------------------
  514. void ctkErrorLogModel::setLogEntryGrouping(bool value)
  515. {
  516. Q_D(ctkErrorLogModel);
  517. d->LogEntryGrouping = value;
  518. }
  519. //------------------------------------------------------------------------------
  520. bool ctkErrorLogModel::asynchronousLogging()const
  521. {
  522. Q_D(const ctkErrorLogModel);
  523. return d->AsynchronousLogging;
  524. }
  525. //------------------------------------------------------------------------------
  526. void ctkErrorLogModel::setAsynchronousLogging(bool value)
  527. {
  528. Q_D(ctkErrorLogModel);
  529. if (d->AsynchronousLogging == value)
  530. {
  531. return;
  532. }
  533. foreach(const QString& handlerName, d->RegisteredHandlers.keys())
  534. {
  535. d->setMessageHandlerConnection(
  536. d->RegisteredHandlers.value(handlerName), value);
  537. }
  538. d->AsynchronousLogging = value;
  539. }
  540. // --------------------------------------------------------------------------
  541. // ctkErrorLogAbstractMessageHandlerPrivate
  542. // --------------------------------------------------------------------------
  543. class ctkErrorLogAbstractMessageHandlerPrivate
  544. {
  545. public:
  546. ctkErrorLogAbstractMessageHandlerPrivate();
  547. ~ctkErrorLogAbstractMessageHandlerPrivate();
  548. bool Enabled;
  549. QString HandlerPrettyName;
  550. // Use "int" instead of "ctkErrorLogModel::TerminalOutput" to avoid compilation warning ...
  551. // qhash.h:879: warning: passing ‘ctkErrorLogModel::TerminalOutput’ chooses ‘int’ over ‘uint’ [-Wsign-promo]
  552. QHash<int, ctkErrorLogTerminalOutput*> TerminalOutputs;
  553. };
  554. // --------------------------------------------------------------------------
  555. ctkErrorLogAbstractMessageHandlerPrivate::
  556. ctkErrorLogAbstractMessageHandlerPrivate()
  557. : Enabled(false)
  558. {
  559. }
  560. // --------------------------------------------------------------------------
  561. ctkErrorLogAbstractMessageHandlerPrivate::~ctkErrorLogAbstractMessageHandlerPrivate()
  562. {
  563. }
  564. // --------------------------------------------------------------------------
  565. // ctkErrorLogAbstractMessageHandlerPrivate methods
  566. // --------------------------------------------------------------------------
  567. ctkErrorLogAbstractMessageHandler::ctkErrorLogAbstractMessageHandler()
  568. : Superclass(), d_ptr(new ctkErrorLogAbstractMessageHandlerPrivate)
  569. {
  570. }
  571. // --------------------------------------------------------------------------
  572. ctkErrorLogAbstractMessageHandler::~ctkErrorLogAbstractMessageHandler()
  573. {
  574. }
  575. // --------------------------------------------------------------------------
  576. QString ctkErrorLogAbstractMessageHandler::handlerPrettyName()const
  577. {
  578. Q_D(const ctkErrorLogAbstractMessageHandler);
  579. if (d->HandlerPrettyName.isEmpty())
  580. {
  581. return this->handlerName();
  582. }
  583. else
  584. {
  585. return d->HandlerPrettyName;
  586. }
  587. }
  588. // --------------------------------------------------------------------------
  589. void ctkErrorLogAbstractMessageHandler::setHandlerPrettyName(const QString& newHandlerPrettyName)
  590. {
  591. Q_D(ctkErrorLogAbstractMessageHandler);
  592. d->HandlerPrettyName = newHandlerPrettyName;
  593. }
  594. // --------------------------------------------------------------------------
  595. bool ctkErrorLogAbstractMessageHandler::enabled()const
  596. {
  597. Q_D(const ctkErrorLogAbstractMessageHandler);
  598. return d->Enabled;
  599. }
  600. // --------------------------------------------------------------------------
  601. void ctkErrorLogAbstractMessageHandler::setEnabled(bool value)
  602. {
  603. Q_D(ctkErrorLogAbstractMessageHandler);
  604. if (value == d->Enabled)
  605. {
  606. return;
  607. }
  608. this->setEnabledInternal(value);
  609. d->Enabled = value;
  610. }
  611. // --------------------------------------------------------------------------
  612. void ctkErrorLogAbstractMessageHandler::handleMessage(const QString& threadId,
  613. ctkErrorLogLevel::LogLevel logLevel,
  614. const QString& origin, const QString& text)
  615. {
  616. Q_D(ctkErrorLogAbstractMessageHandler);
  617. if (logLevel <= ctkErrorLogLevel::Info)
  618. {
  619. if(d->TerminalOutputs.contains(ctkErrorLogModel::StandardOutput))
  620. {
  621. d->TerminalOutputs.value(ctkErrorLogModel::StandardOutput)->output(text);
  622. }
  623. }
  624. else
  625. {
  626. if(d->TerminalOutputs.contains(ctkErrorLogModel::StandardError))
  627. {
  628. d->TerminalOutputs.value(ctkErrorLogModel::StandardError)->output(text);
  629. }
  630. }
  631. emit this->messageHandled(QDateTime::currentDateTime(), threadId, logLevel, origin, text);
  632. }
  633. // --------------------------------------------------------------------------
  634. ctkErrorLogTerminalOutput* ctkErrorLogAbstractMessageHandler::terminalOutput(
  635. ctkErrorLogModel::TerminalOutput terminalOutputType)const
  636. {
  637. Q_D(const ctkErrorLogAbstractMessageHandler);
  638. if(d->TerminalOutputs.contains(terminalOutputType))
  639. {
  640. return d->TerminalOutputs.value(terminalOutputType);
  641. }
  642. return 0;
  643. }
  644. // --------------------------------------------------------------------------
  645. void ctkErrorLogAbstractMessageHandler::setTerminalOutput(
  646. ctkErrorLogModel::TerminalOutput terminalOutputType, ctkErrorLogTerminalOutput* terminalOutput)
  647. {
  648. Q_D(ctkErrorLogAbstractMessageHandler);
  649. d->TerminalOutputs.insert(terminalOutputType, terminalOutput);
  650. }