DebugManager.cpp 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. #include "DebugManager.h"
  2. #include <QApplication>
  3. #include <QDir>
  4. #include <QThread>
  5. #include <QTextStream>
  6. #include <QDateTime>
  7. #include "DialogUserMsg.h"
  8. #include "Preferences.h"
  9. //重定向qdebug输出
  10. void outputLogMessage(QtMsgType type, const QMessageLogContext& context, const QString& msg)
  11. {
  12. //转发给单例的成员函数
  13. DebugManager::getInstance()->outputLog(type, context, msg);
  14. }
  15. DebugManager::DebugManager()
  16. {
  17. //initManager();
  18. }
  19. DebugManager::~DebugManager()
  20. {
  21. freeManager();
  22. }
  23. DebugManager* DebugManager::getInstance()
  24. {
  25. //单例,c++ 11
  26. static DebugManager instance;
  27. return &instance;
  28. }
  29. void DebugManager::outputLog(QtMsgType type, const QMessageLogContext& context, const QString& msg)
  30. {
  31. //如果要写文件需要加锁,因为函数调用在debug调用线程
  32. QMutexLocker locker(&logMutex);
  33. QString out_text;
  34. QTextStream stream(&out_text);
  35. stream << QDateTime::currentDateTime().toString("[yyyy-MM-dd hh:mm:ss]");
  36. //区分日志类型给文本加头
  37. switch (type) {
  38. case QtDebugMsg: stream << "[Debug]________"; break;
  39. case QtInfoMsg: stream << "[Info]_________"; break;
  40. case QtWarningMsg: stream << "[Warning]______"; break;
  41. case QtCriticalMsg: stream << "[Critical]____"; break;
  42. case QtFatalMsg: stream << "[Fatal]_______"; break;
  43. default: stream << "[Unknown]_____"; break;
  44. }
  45. //加个线程id用于测试
  46. stream << msg
  47. // << " ThreadId:"
  48. // << QThread::currentThreadId()
  49. ;
  50. // //写入文件
  51. // if (file.isOpen()) {
  52. // //elapsed距离start的毫秒数
  53. // //这里设置1分钟用来测试
  54. // if (elapsedTimer.elapsed() > 1000 * 60) {
  55. // file.close();
  56. // //重新计时
  57. // elapsedTimer.restart();
  58. // }
  59. // }
  60. // if (!file.isOpen()) {
  61. // //新的文件
  62. // file.setFileName(filePath + QString("/log_%1.txt")
  63. // //.arg(QDateTime::currentDateTime().toString("yyyyMMdd_hhmm")));
  64. // .arg(QDateTime::currentDateTime().toString("yyyyMMdd")));
  65. // //Append追加模式,避免同一文件被清除
  66. // if (!file.open(QIODevice::WriteOnly | QIODevice::Append)) {
  67. // emit newDebug(QtWarningMsg, "Open log file error:" + file.errorString() + file.fileName());
  68. // }
  69. // }
  70. // if (file.isOpen()) {
  71. // //写入文件
  72. // stream.setDevice(&file);
  73. // stream << out_text << endl;
  74. // }
  75. //默认的输出,控制台
  76. //区分日志类型给文本加颜色
  77. //常见格式为:\e[显示方式;前景色;背景色m输出字符串\e[0m
  78. //其中\e=\033
  79. QString cmd_text;
  80. stream.setString(&cmd_text);
  81. // switch (type) {
  82. // //debug绿色
  83. // case QtDebugMsg: stream << "\033[32m"; break;
  84. // //info蓝色
  85. // case QtInfoMsg: stream << "\033[34m"; break;
  86. // //warning黄色
  87. // case QtWarningMsg: stream << "\033[33m"; break;
  88. // //critical黑底白字
  89. // case QtCriticalMsg: stream << "\033[0;37;40m"; break;
  90. // //fatal黑底红字
  91. // //qFatal表示致命错误,默认处理会报异常的
  92. // case QtFatalMsg: stream << "\033[0;31;40m"; break;
  93. // //defualt默认颜色
  94. // default: stream << "\033[0m"; break;
  95. // }
  96. // stream << out_text << "\033[0m";
  97. stream << msg;
  98. defaultOutput(type, context, cmd_text);
  99. QString log_text;
  100. stream.setString(&log_text);
  101. switch (type) {
  102. case QtDebugMsg: stream << "<span style='color:green;'>"; break;
  103. case QtInfoMsg: stream << "<span style='color:blue;'>"; break;
  104. case QtWarningMsg: stream << "<span style='color:yellow;'>"; break;
  105. case QtCriticalMsg: stream << "<span style='color:red;'>"; break;
  106. case QtFatalMsg: stream << "<span style='color:red;'>"; break;
  107. default: stream << "<span style='color:red;'>"; break;
  108. }
  109. stream << out_text << "</span>";
  110. if (type != QtDebugMsg && type != QtInfoMsg )
  111. {
  112. if (g_pDialogUserMsg != nullptr)
  113. {
  114. g_pDialogUserMsg->AddMessage(log_text);
  115. }
  116. }
  117. DEBUGMSG debugMsg;
  118. debugMsg.msgType = type;
  119. debugMsg.strMsg = log_text;
  120. m_AllDebugMsg.push_back(debugMsg);
  121. //发送信号给需要的对象,如ui上显示日志
  122. // emit newDebug(type, log_text);
  123. thePrefs.m_strMessage = out_text;
  124. }
  125. void DebugManager::initManager(const QString& path)
  126. {
  127. //保存路径
  128. filePath = path;
  129. if (filePath.isEmpty())
  130. {
  131. //用到了QApplication::applicationDirPath(),需要先实例化一个app
  132. int argc = 0;
  133. QApplication app(argc, nullptr);
  134. filePath = app.applicationDirPath() + "/log";
  135. }
  136. QDir dir(filePath);
  137. if (!dir.exists())
  138. {
  139. //虽然QFile能够创建不存在的文件,但是它就是不会自动创建不存在的目录
  140. dir.mkpath(filePath);
  141. }
  142. elapsedTimer.start();
  143. //重定向qdebug到自定义函数
  144. defaultOutput = qInstallMessageHandler(outputLogMessage);
  145. }
  146. void DebugManager::freeManager()
  147. {
  148. file.close();
  149. qInstallMessageHandler(nullptr);
  150. }