// NeL - MMORPG Framework // Copyright (C) 2010 Winch Gate Property Limited // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero General Public License as // published by the Free Software Foundation, either version 3 of the // License, or (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . #include "stdmisc.h" #include "nel/misc/path.h" #include "nel/misc/command.h" #include "nel/misc/thread.h" #include "nel/misc/window_displayer.h" using namespace std; #ifdef DEBUG_NEW #define new DEBUG_NEW #endif namespace NLMISC { class CUpdateThread : public IRunnable { CWindowDisplayer *Disp; string WindowNameEx; sint X, Y, W, H, HS; bool Iconified; uint32 FS; string FN; bool WW; CLog *Log; public: CUpdateThread (CWindowDisplayer *disp, string windowNameEx, bool iconified, sint x, sint y, sint w, sint h, sint hs, sint fs, const std::string &fn, bool ww, CLog *log) : Disp(disp), WindowNameEx(windowNameEx), X(x), Y(y), W(w), H(h), HS(hs), Iconified(iconified), FS(fs), FN(fn), WW(ww), Log(log) { } void run() { Disp->open (WindowNameEx, Iconified, X, Y, W, H, HS, FS, FN, WW, Log); Disp->display_main (); } }; CWindowDisplayer::~CWindowDisplayer () { // we have to wait the exit of the thread _Continue = false; nlassert (_Thread != NULL); _Thread->wait(); delete _Thread; } bool CWindowDisplayer::update () { vector copy; { CSynchronized >::CAccessor access (&_CommandsToExecute); copy = access.value(); access.value().clear (); } // execute all commands in the main thread for (uint i = 0; i < copy.size(); i++) { nlassert (Log != NULL); ICommand::execute (copy[i], *Log); } return _Continue; } uint CWindowDisplayer::createLabel (const char *value) { uint pos; { CSynchronized >::CAccessor access (&_Labels); access.value().push_back (CLabelEntry(value)); pos = (uint)access.value().size()-1; } return pos; } void CWindowDisplayer::setLabel (uint label, const string &value) { { CSynchronized >::CAccessor access (&_Labels); nlassert (label < access.value().size()); if (access.value()[label].Value != value) { access.value()[label].Value = value; access.value()[label].NeedUpdate = true; } } } void CWindowDisplayer::create (string windowNameEx, bool iconified, sint x, sint y, sint w, sint h, sint hs, sint fs, const std::string &fn, bool ww, CLog *log) { nlassert (_Thread == NULL); _Thread = IThread::create (new CUpdateThread(this, windowNameEx, iconified, x, y, w, h, hs, fs, fn, ww, log)); Log = log; _Thread->start (); } std::string CWindowDisplayer::stringifyMessage(const NLMISC::CLog::TDisplayInfo &args, const char *message, bool needSlashR) { bool needSpace = false; //stringstream ss; string str; if (args.LogType != CLog::LOG_NO) { str += CWindowDisplayer::logTypeToString(args.LogType); needSpace = true; } // Write thread identifier if (args.ThreadId != 0) { if (needSpace) { str += " "; needSpace = false; } #ifdef NL_OS_WINDOWS str += NLMISC::toString("%4x", args.ThreadId); #else str += NLMISC::toString("%08x", args.ThreadId); #endif needSpace = true; } if (args.FileName != NULL) { if (needSpace) { str += " "; needSpace = false; } str += NLMISC::toString("%20s", CFile::getFilename(args.FileName).c_str()); needSpace = true; } if (args.Line != -1) { if (needSpace) { str += " "; needSpace = false; } str += NLMISC::toString("%4u", args.Line); //ss << setw(4) << args.Line; needSpace = true; } if (args.FuncName != NULL) { if (needSpace) { str += " "; needSpace = false; } str += NLMISC::toString("%20s", args.FuncName); needSpace = true; } if (needSpace) { str += ": "; needSpace = false; } uint nbl = 1; char *npos, *pos = const_cast(message); while ((npos = strchr(pos, '\n'))) { *npos = '\0'; str += pos; if (needSlashR) str += "\r"; str += "\n"; *npos = '\n'; pos = npos + 1; nbl++; } str += pos; pos = const_cast(args.CallstackAndLog.c_str()); while ((npos = strchr(pos, '\n'))) { *npos = '\0'; str += pos; if (needSlashR) str += "\r"; str += "\n"; *npos = '\n'; pos = npos + 1; nbl++; } str += pos; return str; } void CWindowDisplayer::doDisplay (const NLMISC::CLog::TDisplayInfo &args, const char *message) { uint32 color = 0xFF000000; if (args.LogType != CLog::LOG_NO) { if (args.LogType == CLog::LOG_ERROR || args.LogType == CLog::LOG_ASSERT) color = 0x00FF0000; else if (args.LogType == CLog::LOG_WARNING) color = 0x00800000; else if (args.LogType == CLog::LOG_DEBUG) color = 0x00808080; else color = 0; } std::string str = stringifyMessage(args, message, needSlashR); { CSynchronized > >::CAccessor access (&_Buffer); if (_HistorySize > 0 && access.value().size() >= (uint)_HistorySize) { access.value().erase (access.value().begin()); } access.value().push_back (make_pair (color, str)); } } } // NLMISC