#ifndef DEBUG_LOG_H #define DEBUG_LOG_H #include #include #include namespace Debug { enum Level { Error = 1, Warning = 2, Info = 3, Verbose = 4, Marker = Verbose, NoLevel = 5 // Do not filter messages in this case }; extern Level CurrentDebugLevel; } class Log { static std::mutex sLock; std::unique_lock mLock; public: // Locks a global lock while the object is alive Log(Debug::Level level) : mLock(sLock), mLevel(level) { // If the app has no logging system enabled, log level is not specified. // Show all messages without marker - we just use the plain cout in this case. if (Debug::CurrentDebugLevel == Debug::NoLevel) return; if (mLevel <= Debug::CurrentDebugLevel) std::cout << static_cast(mLevel); } // Perfect forwarding wrappers to give the chain of objects to cout template Log& operator<<(T&& rhs) { if (mLevel <= Debug::CurrentDebugLevel) std::cout << std::forward(rhs); return *this; } ~Log() { if (mLevel <= Debug::CurrentDebugLevel) std::cout << std::endl; } private: Debug::Level mLevel; }; #endif