ctkErrorLogStreamMessageHandler.cpp 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206
  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. // CTK includes
  15. #include "ctkErrorLogStreamMessageHandler.h"
  16. // STD includes
  17. #include <iostream>
  18. #include <streambuf>
  19. #include <string>
  20. namespace
  21. {
  22. // --------------------------------------------------------------------------
  23. // ctkStreamHandler
  24. //
  25. // See http://lists.trolltech.com/qt-interest/2005-06/thread00166-0.html
  26. //
  27. // --------------------------------------------------------------------------
  28. class ctkStreamHandler : public std::streambuf
  29. {
  30. public:
  31. ctkStreamHandler(ctkErrorLogStreamMessageHandler* messageHandler,
  32. ctkErrorLogModel::LogLevel logLevel,
  33. std::ostream& stream);
  34. void setEnabled(bool value);
  35. protected:
  36. virtual int_type overflow(int_type v);
  37. virtual std::streamsize xsputn(const char *p, std::streamsize n);
  38. private:
  39. ctkErrorLogStreamMessageHandler * MessageHandler;
  40. ctkErrorLogModel::LogLevel LogLevel;
  41. bool Enabled;
  42. std::ostream& Stream;
  43. std::streambuf* SavedBuffer;
  44. std::string StringBuffer;
  45. };
  46. // --------------------------------------------------------------------------
  47. // ctkStreamHandler methods
  48. // --------------------------------------------------------------------------
  49. ctkStreamHandler::ctkStreamHandler(ctkErrorLogStreamMessageHandler* messageHandler,
  50. ctkErrorLogModel::LogLevel logLevel,
  51. std::ostream& stream) :
  52. MessageHandler(messageHandler), LogLevel(logLevel), Stream(stream)
  53. {
  54. this->Enabled = false;
  55. }
  56. // --------------------------------------------------------------------------
  57. void ctkStreamHandler::setEnabled(bool value)
  58. {
  59. if (this->Enabled == value)
  60. {
  61. return;
  62. }
  63. if (value)
  64. {
  65. this->SavedBuffer = this->Stream.rdbuf();
  66. this->Stream.rdbuf(this);
  67. }
  68. else
  69. {
  70. // Output anything that is left
  71. if (!this->StringBuffer.empty())
  72. {
  73. Q_ASSERT(this->MessageHandler->errorLogModel());
  74. this->MessageHandler->errorLogModel()->addEntry(
  75. this->LogLevel, this->MessageHandler->handlerPrettyName(), this->StringBuffer.c_str());
  76. }
  77. this->Stream.rdbuf(this->SavedBuffer);
  78. }
  79. this->Enabled = value;
  80. }
  81. // --------------------------------------------------------------------------
  82. std::streambuf::int_type ctkStreamHandler::overflow(std::streambuf::int_type v)
  83. {
  84. if (v == '\n')
  85. {
  86. Q_ASSERT(this->MessageHandler->errorLogModel());
  87. this->MessageHandler->errorLogModel()->addEntry(
  88. this->LogLevel, this->MessageHandler->handlerPrettyName(), this->StringBuffer.c_str());
  89. this->StringBuffer.erase(this->StringBuffer.begin(), this->StringBuffer.end());
  90. }
  91. else
  92. {
  93. this->StringBuffer += v;
  94. }
  95. return v;
  96. }
  97. // --------------------------------------------------------------------------
  98. std::streamsize ctkStreamHandler::xsputn(const char *p, std::streamsize n)
  99. {
  100. this->StringBuffer.append(p, p + n);
  101. std::string::size_type pos = 0;
  102. while (pos != std::string::npos)
  103. {
  104. pos = this->StringBuffer.find('\n');
  105. if (pos != std::string::npos)
  106. {
  107. std::string tmp(this->StringBuffer.begin(), this->StringBuffer.begin() + pos);
  108. Q_ASSERT(this->MessageHandler->errorLogModel());
  109. this->MessageHandler->errorLogModel()->addEntry(
  110. this->LogLevel, this->MessageHandler->handlerPrettyName(), tmp.c_str());
  111. this->StringBuffer.erase(this->StringBuffer.begin(), this->StringBuffer.begin() + pos + 1);
  112. }
  113. }
  114. return n;
  115. }
  116. }
  117. // --------------------------------------------------------------------------
  118. // ctkErrorLogStreamMessageHandlerPrivate
  119. // --------------------------------------------------------------------------
  120. class ctkErrorLogStreamMessageHandlerPrivate
  121. {
  122. public:
  123. ctkErrorLogStreamMessageHandlerPrivate();
  124. ~ctkErrorLogStreamMessageHandlerPrivate();
  125. ctkStreamHandler * CoutStreamHandler;
  126. ctkStreamHandler * CerrStreamHandler;
  127. };
  128. // --------------------------------------------------------------------------
  129. // ctkErrorLogStreamMessageHandlerPrivate methods
  130. //------------------------------------------------------------------------------
  131. ctkErrorLogStreamMessageHandlerPrivate::ctkErrorLogStreamMessageHandlerPrivate()
  132. {
  133. }
  134. //------------------------------------------------------------------------------
  135. ctkErrorLogStreamMessageHandlerPrivate::~ctkErrorLogStreamMessageHandlerPrivate()
  136. {
  137. delete this->CoutStreamHandler;
  138. delete this->CerrStreamHandler;
  139. }
  140. //------------------------------------------------------------------------------
  141. // ctkErrorLogStreamMessageHandler methods
  142. //------------------------------------------------------------------------------
  143. QString ctkErrorLogStreamMessageHandler::HandlerName = QLatin1String("Stream");
  144. // --------------------------------------------------------------------------
  145. ctkErrorLogStreamMessageHandler::ctkErrorLogStreamMessageHandler() :
  146. Superclass(), d_ptr(new ctkErrorLogStreamMessageHandlerPrivate())
  147. {
  148. Q_D(ctkErrorLogStreamMessageHandler);
  149. d->CoutStreamHandler = new ctkStreamHandler(this, ctkErrorLogModel::Info, std::cout);
  150. d->CerrStreamHandler = new ctkStreamHandler(this, ctkErrorLogModel::Critical, std::cerr);
  151. }
  152. // --------------------------------------------------------------------------
  153. ctkErrorLogStreamMessageHandler::~ctkErrorLogStreamMessageHandler()
  154. {
  155. }
  156. // --------------------------------------------------------------------------
  157. QString ctkErrorLogStreamMessageHandler::handlerName()const
  158. {
  159. return ctkErrorLogStreamMessageHandler::HandlerName;
  160. }
  161. // --------------------------------------------------------------------------
  162. void ctkErrorLogStreamMessageHandler::setEnabledInternal(bool value)
  163. {
  164. Q_D(ctkErrorLogStreamMessageHandler);
  165. d->CoutStreamHandler->setEnabled(value);
  166. d->CerrStreamHandler->setEnabled(value);
  167. }