| 
									
										
										
										
											2019-03-14 17:55:06 +08:00
										 |  |  | // Copyright (c) 2016-2018 Easy2D - Nomango
 | 
					
						
							|  |  |  | // 
 | 
					
						
							|  |  |  | // Permission is hereby granted, free of charge, to any person obtaining a copy
 | 
					
						
							|  |  |  | // of this software and associated documentation files (the "Software"), to deal
 | 
					
						
							|  |  |  | // in the Software without restriction, including without limitation the rights
 | 
					
						
							|  |  |  | // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 | 
					
						
							|  |  |  | // copies of the Software, and to permit persons to whom the Software is
 | 
					
						
							|  |  |  | // furnished to do so, subject to the following conditions:
 | 
					
						
							|  |  |  | // 
 | 
					
						
							|  |  |  | // The above copyright notice and this permission notice shall be included in
 | 
					
						
							|  |  |  | // all copies or substantial portions of the Software.
 | 
					
						
							|  |  |  | // 
 | 
					
						
							|  |  |  | // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | 
					
						
							|  |  |  | // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | 
					
						
							|  |  |  | // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 | 
					
						
							|  |  |  | // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | 
					
						
							|  |  |  | // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 | 
					
						
							|  |  |  | // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 | 
					
						
							|  |  |  | // THE SOFTWARE.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | #include "../macros.h"
 | 
					
						
							|  |  |  | #include "../common/Singleton.hpp"
 | 
					
						
							|  |  |  | #include <ctime>
 | 
					
						
							|  |  |  | #include <iomanip>
 | 
					
						
							|  |  |  | #include <iostream>
 | 
					
						
							|  |  |  | #include <sstream>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef E2D_LOG
 | 
					
						
							|  |  |  | #	ifdef E2D_DEBUG
 | 
					
						
							|  |  |  | #		define E2D_LOG(FORMAT, ...) easy2d::Logger::Instance().Messageln((FORMAT), __VA_ARGS__)
 | 
					
						
							|  |  |  | #	else
 | 
					
						
							|  |  |  | #		define E2D_LOG __noop
 | 
					
						
							|  |  |  | #	endif
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef E2D_WARNING_LOG
 | 
					
						
							|  |  |  | #	define E2D_WARNING_LOG(FORMAT, ...) easy2d::Logger::Instance().Warningln((FORMAT), __VA_ARGS__)
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef E2D_ERROR_LOG
 | 
					
						
							|  |  |  | #	define E2D_ERROR_LOG(FORMAT, ...) easy2d::Logger::Instance().Errorln((FORMAT), __VA_ARGS__)
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace easy2d | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	namespace __console_colors | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		using ConsoleColor = std::wostream&(*)(std::wostream&); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define DECLARE_COLOR(COLOR)\
 | 
					
						
							|  |  |  | 		extern std::wostream&(stdout_##COLOR)(std::wostream&);\ | 
					
						
							|  |  |  | 		extern std::wostream&(stderr_##COLOR)(std::wostream&); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define DECLARE_BG_COLOR(COLOR)\
 | 
					
						
							|  |  |  | 		extern std::wostream&(stdout_##COLOR##_bg)(std::wostream&);\ | 
					
						
							|  |  |  | 		extern std::wostream&(stderr_##COLOR##_bg)(std::wostream&); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		DECLARE_COLOR(red); | 
					
						
							|  |  |  | 		DECLARE_COLOR(green); | 
					
						
							|  |  |  | 		DECLARE_COLOR(yellow); | 
					
						
							|  |  |  | 		DECLARE_COLOR(blue); | 
					
						
							|  |  |  | 		DECLARE_COLOR(white); | 
					
						
							|  |  |  | 		DECLARE_COLOR(reset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		DECLARE_BG_COLOR(red); | 
					
						
							|  |  |  | 		DECLARE_BG_COLOR(green); | 
					
						
							|  |  |  | 		DECLARE_BG_COLOR(yellow); | 
					
						
							|  |  |  | 		DECLARE_BG_COLOR(blue); | 
					
						
							|  |  |  | 		DECLARE_BG_COLOR(white); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #undef DECLARE_COLOR
 | 
					
						
							|  |  |  | #undef DECLARE_BG_COLOR
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	class E2D_API Logger | 
					
						
							|  |  |  | 		: public Singleton<Logger> | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		E2D_DECLARE_SINGLETON(Logger); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	public: | 
					
						
							|  |  |  | 		inline void Enable() | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			enabled_ = true; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		inline void Disable() | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			enabled_ = false; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		template <typename ..._Args> | 
					
						
							|  |  |  | 		inline void Print(const wchar_t* format, _Args&&... args) const | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			using namespace __console_colors; | 
					
						
							| 
									
										
										
										
											2019-03-21 14:14:11 +08:00
										 |  |  | 			Output(std::wcout, stdout_white, nullptr, format, std::forward<_Args>(args)...); | 
					
						
							| 
									
										
										
										
											2019-03-14 17:55:06 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		template <typename ..._Args> | 
					
						
							|  |  |  | 		inline void Println(const wchar_t* format, _Args&&... args) const | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			using namespace __console_colors; | 
					
						
							| 
									
										
										
										
											2019-03-21 14:14:11 +08:00
										 |  |  | 			OutputLine(std::wcout, stdout_white, nullptr, format, std::forward<_Args>(args)...); | 
					
						
							| 
									
										
										
										
											2019-03-14 17:55:06 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		template <typename ..._Args> | 
					
						
							|  |  |  | 		inline void Message(const wchar_t * format, _Args&&... args) const | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			using namespace __console_colors; | 
					
						
							| 
									
										
										
										
											2019-03-21 14:14:11 +08:00
										 |  |  | 			Output(std::wcout, stdout_blue, nullptr, format, std::forward<_Args>(args)...); | 
					
						
							| 
									
										
										
										
											2019-03-14 17:55:06 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		template <typename ..._Args> | 
					
						
							|  |  |  | 		inline void Messageln(const wchar_t * format, _Args&&... args) const | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			using namespace __console_colors; | 
					
						
							| 
									
										
										
										
											2019-03-21 14:14:11 +08:00
										 |  |  | 			OutputLine(std::wcout, stdout_blue, nullptr, format, std::forward<_Args>(args)...); | 
					
						
							| 
									
										
										
										
											2019-03-14 17:55:06 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		template <typename ..._Args> | 
					
						
							|  |  |  | 		inline void Warning(const wchar_t* format, _Args&&... args) const | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			using namespace __console_colors; | 
					
						
							| 
									
										
										
										
											2019-03-21 14:14:11 +08:00
										 |  |  | 			Output(std::wcerr, stdout_yellow_bg, L"Warning: ", format, std::forward<_Args>(args)...); | 
					
						
							| 
									
										
										
										
											2019-03-14 17:55:06 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		template <typename ..._Args> | 
					
						
							|  |  |  | 		inline void Warningln(const wchar_t* format, _Args&&... args) const | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			using namespace __console_colors; | 
					
						
							| 
									
										
										
										
											2019-03-21 14:14:11 +08:00
										 |  |  | 			OutputLine(std::wcerr, stdout_yellow_bg, L"Warning: ", format, std::forward<_Args>(args)...); | 
					
						
							| 
									
										
										
										
											2019-03-14 17:55:06 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		template <typename ..._Args> | 
					
						
							|  |  |  | 		inline void Error(const wchar_t* format, _Args&&... args) const | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			using namespace __console_colors; | 
					
						
							| 
									
										
										
										
											2019-03-21 14:14:11 +08:00
										 |  |  | 			Output(std::wcerr, stderr_red_bg, L"Error: ", format, std::forward<_Args>(args)...); | 
					
						
							| 
									
										
										
										
											2019-03-14 17:55:06 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		template <typename ..._Args> | 
					
						
							|  |  |  | 		inline void Errorln(const wchar_t* format, _Args&&... args) const | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			using namespace __console_colors; | 
					
						
							| 
									
										
										
										
											2019-03-21 14:14:11 +08:00
										 |  |  | 			OutputLine(std::wcerr, stderr_red_bg, L"Error: ", format, std::forward<_Args>(args)...); | 
					
						
							| 
									
										
										
										
											2019-03-14 17:55:06 +08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	private: | 
					
						
							|  |  |  | 		Logger(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		template <typename ..._Args> | 
					
						
							|  |  |  | 		inline void OutputLine(std::wostream& os, __console_colors::ConsoleColor color, const wchar_t* prompt, const wchar_t* format, _Args&&... args) const | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if (!enabled_) | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-21 14:14:11 +08:00
										 |  |  | 			Output(os, color, prompt, format, std::forward<_Args>(args)...); | 
					
						
							| 
									
										
										
										
											2019-03-14 17:55:06 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			os << std::endl; | 
					
						
							|  |  |  | 			::OutputDebugStringW(L"\r\n"); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		template <typename ..._Args> | 
					
						
							|  |  |  | 		inline void Output(std::wostream& os, __console_colors::ConsoleColor color, const wchar_t* prompt, const wchar_t* format, _Args&&... args) const | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if (!enabled_) | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-21 14:14:11 +08:00
										 |  |  | 			std::wstring output = MakeOutputString(prompt, format, std::forward<_Args>(args)...); | 
					
						
							| 
									
										
										
										
											2019-03-14 17:55:06 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			os << color << output; | 
					
						
							|  |  |  | 			::OutputDebugStringW(output.c_str()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			ResetColor(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		template <typename ..._Args> | 
					
						
							|  |  |  | 		inline std::wstring MakeOutputString(const wchar_t* prompt, const wchar_t* format, _Args&&... args) const | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			static wchar_t temp_buffer[1024 * 3 + 1]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-21 14:14:11 +08:00
										 |  |  | 			const auto len = ::_scwprintf(format, std::forward<_Args>(args)...); | 
					
						
							|  |  |  | 			::swprintf_s(temp_buffer, len + 1, format, std::forward<_Args>(args)...); | 
					
						
							| 
									
										
										
										
											2019-03-14 17:55:06 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			std::wstringstream ss; | 
					
						
							|  |  |  | 			ss << Logger::OutPrefix; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (prompt) | 
					
						
							|  |  |  | 				ss << prompt; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			ss << temp_buffer; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			return ss.str(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		inline void ResetColor() const | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			::SetConsoleTextAttribute(::GetStdHandle(STD_OUTPUT_HANDLE), default_stdout_color_); | 
					
						
							|  |  |  | 			::SetConsoleTextAttribute(::GetStdHandle(STD_ERROR_HANDLE), default_stderr_color_); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		static inline std::wostream& OutPrefix(std::wostream& out) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			std::time_t unix = std::time(nullptr); | 
					
						
							|  |  |  | 			std::tm tmbuf; | 
					
						
							|  |  |  | 			localtime_s(&tmbuf, &unix); | 
					
						
							|  |  |  | 			out << std::put_time(&tmbuf, L"[easy2d] %H:%M:%S "); | 
					
						
							|  |  |  | 			return out; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	private: | 
					
						
							|  |  |  | 		bool enabled_; | 
					
						
							|  |  |  | 		WORD default_stdout_color_; | 
					
						
							|  |  |  | 		WORD default_stderr_color_; | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace easy2d | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	inline void ThrowIfFailed(HRESULT hr) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		if (FAILED(hr)) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			E2D_ERROR_LOG(L"Fatal error with HRESULT of %08X", hr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			static char buffer[1024 + 1]; | 
					
						
							|  |  |  | 			sprintf_s(buffer, "Fatal error with HRESULT of %08X", hr); | 
					
						
							|  |  |  | 			throw std::runtime_error(buffer); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |