| 
									
										
										
										
											2018-11-08 21:39:26 +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.
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #include "render.h"
 | 
					
						
							|  |  |  |  | #include "time.h"
 | 
					
						
							|  |  |  |  | #include "base.h"
 | 
					
						
							| 
									
										
										
										
											2018-11-15 17:59:18 +08:00
										 |  |  |  | #include "logs.h"
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | #include "modules.h"
 | 
					
						
							| 
									
										
										
										
											2018-11-14 01:34:41 +08:00
										 |  |  |  | #include "Image.h"
 | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | #pragma comment(lib, "d2d1.lib")
 | 
					
						
							|  |  |  |  | #pragma comment(lib, "dwrite.lib")
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | namespace easy2d | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 	namespace devices | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 		GraphicsDevice::GraphicsDevice() | 
					
						
							|  |  |  |  | 			: fps_text_format_(nullptr) | 
					
						
							|  |  |  |  | 			, fps_text_layout_(nullptr) | 
					
						
							|  |  |  |  | 			, clear_color_(D2D1::ColorF(D2D1::ColorF::Black)) | 
					
						
							| 
									
										
										
										
											2018-11-12 22:36:50 +08:00
										 |  |  |  | 			, initialized(false) | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			ZeroMemory(&d2d, sizeof(D2DResources)); | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-14 01:34:41 +08:00
										 |  |  |  | 			modules::Init(); | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 		GraphicsDevice::~GraphicsDevice() | 
					
						
							|  |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-15 17:59:18 +08:00
										 |  |  |  | 			E2D_LOG("Destroying graphics device"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			ClearImageCache(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			SafeRelease(fps_text_format_); | 
					
						
							|  |  |  |  | 			SafeRelease(fps_text_layout_); | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 			SafeRelease(d2d.text_renderer); | 
					
						
							|  |  |  |  | 			SafeRelease(d2d.solid_brush); | 
					
						
							|  |  |  |  | 			SafeRelease(d2d.render_target); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			SafeRelease(d2d.miter_stroke_style); | 
					
						
							|  |  |  |  | 			SafeRelease(d2d.bevel_stroke_style); | 
					
						
							|  |  |  |  | 			SafeRelease(d2d.round_stroke_style); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			SafeRelease(d2d.imaging_factory); | 
					
						
							|  |  |  |  | 			SafeRelease(d2d.write_factory); | 
					
						
							|  |  |  |  | 			SafeRelease(d2d.factory); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			modules::Destroy(); | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-14 01:34:41 +08:00
										 |  |  |  | 		void GraphicsDevice::Init(HWND hwnd, bool debug) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 22:36:50 +08:00
										 |  |  |  | 			if (initialized) | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 				return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-15 17:59:18 +08:00
										 |  |  |  | 			E2D_LOG("Initing graphics device"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 22:36:50 +08:00
										 |  |  |  | 			D2D1_FACTORY_OPTIONS options{ debug ? D2D1_DEBUG_LEVEL_INFORMATION : D2D1_DEBUG_LEVEL_NONE }; | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 			ThrowIfFailed( | 
					
						
							|  |  |  |  | 				D2D1CreateFactory( | 
					
						
							|  |  |  |  | 					D2D1_FACTORY_TYPE_SINGLE_THREADED, | 
					
						
							| 
									
										
										
										
											2018-11-12 22:36:50 +08:00
										 |  |  |  | 					__uuidof(ID2D1Factory), | 
					
						
							|  |  |  |  | 					&options, | 
					
						
							|  |  |  |  | 					reinterpret_cast<void**>(&d2d.factory) | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 				) | 
					
						
							|  |  |  |  | 			); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			ThrowIfFailed( | 
					
						
							|  |  |  |  | 				CoCreateInstance( | 
					
						
							|  |  |  |  | 					CLSID_WICImagingFactory, | 
					
						
							|  |  |  |  | 					nullptr, | 
					
						
							|  |  |  |  | 					CLSCTX_INPROC_SERVER, | 
					
						
							| 
									
										
										
										
											2018-11-12 22:36:50 +08:00
										 |  |  |  | 					__uuidof(IWICImagingFactory), | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 					reinterpret_cast<void**>(&d2d.imaging_factory) | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 				) | 
					
						
							|  |  |  |  | 			); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			ThrowIfFailed( | 
					
						
							|  |  |  |  | 				DWriteCreateFactory( | 
					
						
							|  |  |  |  | 					DWRITE_FACTORY_TYPE_SHARED, | 
					
						
							|  |  |  |  | 					__uuidof(IDWriteFactory), | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 					reinterpret_cast<IUnknown**>(&d2d.write_factory) | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 				) | 
					
						
							|  |  |  |  | 			); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			auto stroke_style = D2D1::StrokeStyleProperties( | 
					
						
							|  |  |  |  | 				D2D1_CAP_STYLE_FLAT, | 
					
						
							|  |  |  |  | 				D2D1_CAP_STYLE_FLAT, | 
					
						
							|  |  |  |  | 				D2D1_CAP_STYLE_FLAT, | 
					
						
							|  |  |  |  | 				D2D1_LINE_JOIN_MITER, | 
					
						
							|  |  |  |  | 				2.0f, | 
					
						
							|  |  |  |  | 				D2D1_DASH_STYLE_SOLID, | 
					
						
							|  |  |  |  | 				0.0f | 
					
						
							|  |  |  |  | 			); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			ThrowIfFailed( | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 				d2d.factory->CreateStrokeStyle( | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 					stroke_style, | 
					
						
							|  |  |  |  | 					nullptr, | 
					
						
							|  |  |  |  | 					0, | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 					&d2d.miter_stroke_style | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 				) | 
					
						
							|  |  |  |  | 			); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			stroke_style.lineJoin = D2D1_LINE_JOIN_BEVEL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			ThrowIfFailed( | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 				d2d.factory->CreateStrokeStyle( | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 					stroke_style, | 
					
						
							|  |  |  |  | 					nullptr, | 
					
						
							|  |  |  |  | 					0, | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 					&d2d.bevel_stroke_style | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 				) | 
					
						
							|  |  |  |  | 			); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			stroke_style.lineJoin = D2D1_LINE_JOIN_ROUND; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			ThrowIfFailed( | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 				d2d.factory->CreateStrokeStyle( | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 					stroke_style, | 
					
						
							|  |  |  |  | 					nullptr, | 
					
						
							|  |  |  |  | 					0, | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 					&d2d.round_stroke_style | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 				) | 
					
						
							|  |  |  |  | 			); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			CreateDeviceResources(hwnd); | 
					
						
							| 
									
										
										
										
											2018-11-12 22:36:50 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 			initialized = true; | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 		void GraphicsDevice::BeginDraw(HWND hwnd) | 
					
						
							|  |  |  |  | 		{ | 
					
						
							|  |  |  |  | 			CreateDeviceResources(hwnd); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			d2d.render_target->BeginDraw(); | 
					
						
							|  |  |  |  | 			d2d.render_target->Clear(clear_color_); | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 		void GraphicsDevice::EndDraw() | 
					
						
							|  |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			HRESULT hr = d2d.render_target->EndDraw(); | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 			if (hr == D2DERR_RECREATE_TARGET) | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 			{ | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				// <20><><EFBFBD><EFBFBD> Direct3D <20>豸<EFBFBD><E8B1B8>ִ<EFBFBD>й<EFBFBD><D0B9><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ʧ<EFBFBD><CAA7><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ǰ<EFBFBD><C7B0><EFBFBD>豸<EFBFBD><E8B1B8><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Դ
 | 
					
						
							|  |  |  |  | 				// <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD>һ<EFBFBD>ε<EFBFBD><CEB5><EFBFBD>ʱ<EFBFBD>ؽ<EFBFBD><D8BD><EFBFBD>Դ
 | 
					
						
							|  |  |  |  | 				hr = S_OK; | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				SafeRelease(fps_text_format_); | 
					
						
							|  |  |  |  | 				SafeRelease(fps_text_layout_); | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 				SafeRelease(d2d.text_renderer); | 
					
						
							|  |  |  |  | 				SafeRelease(d2d.solid_brush); | 
					
						
							|  |  |  |  | 				SafeRelease(d2d.render_target); | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			ThrowIfFailed(hr); | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 		void GraphicsDevice::ClearImageCache() | 
					
						
							|  |  |  |  | 		{ | 
					
						
							|  |  |  |  | 			if (bitmap_cache_.empty()) | 
					
						
							|  |  |  |  | 				return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			for (const auto& bitmap : bitmap_cache_) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				bitmap.second->Release(); | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 			bitmap_cache_.clear(); | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 		HRESULT GraphicsDevice::CreateRectGeometry( | 
					
						
							|  |  |  |  | 			const math::Matrix& matrix, | 
					
						
							|  |  |  |  | 			const Size& size, | 
					
						
							|  |  |  |  | 			ID2D1Geometry** geometry | 
					
						
							|  |  |  |  | 		) const | 
					
						
							|  |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (!d2d.factory) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				return E_UNEXPECTED; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			HRESULT hr; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			ID2D1RectangleGeometry * rectangle = nullptr; | 
					
						
							|  |  |  |  | 			ID2D1TransformedGeometry * transformed = nullptr; | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 			hr = d2d.factory->CreateRectangleGeometry( | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				D2D1::RectF(0, 0, size.width, size.height), | 
					
						
							|  |  |  |  | 				&rectangle | 
					
						
							|  |  |  |  | 			); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			if (SUCCEEDED(hr)) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 				hr = d2d.factory->CreateTransformedGeometry( | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 					rectangle, | 
					
						
							|  |  |  |  | 					matrix, | 
					
						
							|  |  |  |  | 					&transformed | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 				); | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			if (SUCCEEDED(hr)) | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 			{ | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				*geometry = transformed; | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			SafeRelease(rectangle); | 
					
						
							|  |  |  |  | 			return hr; | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 		HRESULT GraphicsDevice::CreateTextFormat(IDWriteTextFormat ** text_format, const Font & font) const | 
					
						
							|  |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (!d2d.write_factory) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				return E_UNEXPECTED; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			return d2d.write_factory->CreateTextFormat( | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				font.family.c_str(), | 
					
						
							|  |  |  |  | 				nullptr, | 
					
						
							|  |  |  |  | 				DWRITE_FONT_WEIGHT(font.weight), | 
					
						
							|  |  |  |  | 				font.italic ? DWRITE_FONT_STYLE_ITALIC : DWRITE_FONT_STYLE_NORMAL, | 
					
						
							|  |  |  |  | 				DWRITE_FONT_STRETCH_NORMAL, | 
					
						
							|  |  |  |  | 				font.size, | 
					
						
							|  |  |  |  | 				L"", | 
					
						
							|  |  |  |  | 				text_format | 
					
						
							|  |  |  |  | 			); | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 		HRESULT GraphicsDevice::CreateTextLayout(IDWriteTextLayout ** text_layout, const String & text, IDWriteTextFormat * text_format, float wrap_width) const | 
					
						
							|  |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (!d2d.write_factory) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				return E_UNEXPECTED; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			UINT32 length = static_cast<UINT32>(text.length()); | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			return d2d.write_factory->CreateTextLayout( | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				text.c_str(), | 
					
						
							|  |  |  |  | 				length, | 
					
						
							|  |  |  |  | 				text_format, | 
					
						
							|  |  |  |  | 				wrap_width, | 
					
						
							|  |  |  |  | 				0, | 
					
						
							|  |  |  |  | 				text_layout | 
					
						
							|  |  |  |  | 			); | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 		HRESULT GraphicsDevice::CreateLayer(ID2D1Layer ** layer) | 
					
						
							|  |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (!d2d.render_target) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				return E_UNEXPECTED; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			return d2d.render_target->CreateLayer(layer); | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 		HRESULT GraphicsDevice::DrawGeometry( | 
					
						
							|  |  |  |  | 			ID2D1Geometry * geometry, | 
					
						
							|  |  |  |  | 			const Color & border_color, | 
					
						
							|  |  |  |  | 			float opacity, | 
					
						
							|  |  |  |  | 			float stroke_width, | 
					
						
							|  |  |  |  | 			StrokeStyle stroke | 
					
						
							|  |  |  |  | 		) | 
					
						
							|  |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (!d2d.solid_brush || | 
					
						
							|  |  |  |  | 				!d2d.render_target) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				return E_UNEXPECTED; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			d2d.solid_brush->SetColor(border_color); | 
					
						
							|  |  |  |  | 			d2d.solid_brush->SetOpacity(opacity); | 
					
						
							|  |  |  |  | 			d2d.render_target->DrawGeometry( | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				geometry, | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 				d2d.solid_brush, | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				stroke_width, | 
					
						
							|  |  |  |  | 				GetStrokeStyle(stroke) | 
					
						
							|  |  |  |  | 			); | 
					
						
							|  |  |  |  | 			return S_OK; | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 		ID2D1StrokeStyle * GraphicsDevice::GetStrokeStyle(StrokeStyle stroke) const | 
					
						
							|  |  |  |  | 		{ | 
					
						
							|  |  |  |  | 			ID2D1StrokeStyle * stroke_style = nullptr; | 
					
						
							|  |  |  |  | 			switch (stroke) | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 			{ | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			case StrokeStyle::Miter: | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 				stroke_style = d2d.miter_stroke_style; | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				break; | 
					
						
							|  |  |  |  | 			case StrokeStyle::Bevel: | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 				stroke_style = d2d.bevel_stroke_style; | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				break; | 
					
						
							|  |  |  |  | 			case StrokeStyle::Round: | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 				stroke_style = d2d.round_stroke_style; | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				break; | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			return stroke_style; | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 		HRESULT GraphicsDevice::DrawImage( | 
					
						
							| 
									
										
										
										
											2018-11-14 01:34:41 +08:00
										 |  |  |  | 			spImage const& image, | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			float opacity, | 
					
						
							|  |  |  |  | 			const Rect & dest_rect, | 
					
						
							|  |  |  |  | 			const Rect & source_rect | 
					
						
							|  |  |  |  | 		) | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (!d2d.render_target) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				return E_UNEXPECTED; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			d2d.render_target->DrawBitmap( | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				image->GetBitmap(), | 
					
						
							|  |  |  |  | 				dest_rect, | 
					
						
							|  |  |  |  | 				opacity, | 
					
						
							|  |  |  |  | 				D2D1_BITMAP_INTERPOLATION_MODE_LINEAR, | 
					
						
							|  |  |  |  | 				source_rect | 
					
						
							|  |  |  |  | 			); | 
					
						
							|  |  |  |  | 			return S_OK; | 
					
						
							|  |  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 		HRESULT GraphicsDevice::DrawTextLayout(IDWriteTextLayout * text_layout) | 
					
						
							|  |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (!d2d.text_renderer) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				return E_UNEXPECTED; | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			return text_layout->Draw(nullptr, d2d.text_renderer, 0, 0); | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 		HRESULT GraphicsDevice::PushClip(const math::Matrix & clip_matrix, const Size & clip_size) | 
					
						
							|  |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (!d2d.render_target) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				return E_UNEXPECTED; | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			d2d.render_target->SetTransform(clip_matrix); | 
					
						
							|  |  |  |  | 			d2d.render_target->PushAxisAlignedClip( | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				D2D1::RectF(0, 0, clip_size.width, clip_size.height), | 
					
						
							|  |  |  |  | 				D2D1_ANTIALIAS_MODE_PER_PRIMITIVE | 
					
						
							|  |  |  |  | 			); | 
					
						
							|  |  |  |  | 			return S_OK; | 
					
						
							|  |  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 		HRESULT GraphicsDevice::PopClip() | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (!d2d.render_target) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				return E_UNEXPECTED; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			d2d.render_target->PopAxisAlignedClip(); | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			return S_OK; | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 		HRESULT GraphicsDevice::PushLayer(ID2D1Layer * layer, LayerProperties properties) | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (!d2d.render_target || | 
					
						
							|  |  |  |  | 				!d2d.solid_brush) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				return E_UNEXPECTED; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			d2d.render_target->PushLayer( | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				D2D1::LayerParameters( | 
					
						
							|  |  |  |  | 					properties.area, | 
					
						
							|  |  |  |  | 					nullptr, | 
					
						
							|  |  |  |  | 					D2D1_ANTIALIAS_MODE_PER_PRIMITIVE, | 
					
						
							|  |  |  |  | 					D2D1::Matrix3x2F::Identity(), | 
					
						
							|  |  |  |  | 					properties.opacity, | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 					d2d.solid_brush, | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 					D2D1_LAYER_OPTIONS_NONE | 
					
						
							|  |  |  |  | 				), | 
					
						
							|  |  |  |  | 				layer | 
					
						
							|  |  |  |  | 			); | 
					
						
							|  |  |  |  | 			return S_OK; | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 		HRESULT GraphicsDevice::PopLayer() | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (!d2d.render_target) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				return E_UNEXPECTED; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			d2d.render_target->PopLayer(); | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			return S_OK; | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 		HRESULT GraphicsDevice::CreateBitmapFromFile(const String & file_path, ID2D1Bitmap ** bitmap) | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (d2d.imaging_factory == nullptr || | 
					
						
							|  |  |  |  | 				d2d.render_target == nullptr) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				return E_UNEXPECTED; | 
					
						
							|  |  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			if (bitmap == nullptr) | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 			{ | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				return E_POINTER; | 
					
						
							|  |  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			size_t hash_code = std::hash<String>{}(file_path); | 
					
						
							|  |  |  |  | 			if (bitmap_cache_.find(hash_code) != bitmap_cache_.end()) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				*bitmap = bitmap_cache_[hash_code]; | 
					
						
							|  |  |  |  | 				return S_OK; | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			IWICBitmapDecoder*		decoder = nullptr; | 
					
						
							|  |  |  |  | 			IWICBitmapFrameDecode*	source = nullptr; | 
					
						
							|  |  |  |  | 			IWICStream*				stream = nullptr; | 
					
						
							|  |  |  |  | 			IWICFormatConverter*	converter = nullptr; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			// <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			HRESULT hr = d2d.imaging_factory->CreateDecoderFromFilename( | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				file_path.c_str(), | 
					
						
							|  |  |  |  | 				nullptr, | 
					
						
							|  |  |  |  | 				GENERIC_READ, | 
					
						
							|  |  |  |  | 				WICDecodeMetadataCacheOnLoad, | 
					
						
							|  |  |  |  | 				&decoder | 
					
						
							|  |  |  |  | 			); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			if (SUCCEEDED(hr)) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				// <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ʼ<EFBFBD><CABC><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
 | 
					
						
							|  |  |  |  | 				hr = decoder->GetFrame(0, &source); | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			if (SUCCEEDED(hr)) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				// <20><><EFBFBD><EFBFBD>ͼƬ<CDBC><C6AC>ʽת<CABD><D7AA><EFBFBD><EFBFBD>
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 				hr = d2d.imaging_factory->CreateFormatConverter(&converter); | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			if (SUCCEEDED(hr)) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				// ͼƬ<CDBC><C6AC>ʽת<CABD><D7AA><EFBFBD><EFBFBD> 32bppPBGRA
 | 
					
						
							|  |  |  |  | 				hr = converter->Initialize( | 
					
						
							|  |  |  |  | 					source, | 
					
						
							|  |  |  |  | 					GUID_WICPixelFormat32bppPBGRA, | 
					
						
							|  |  |  |  | 					WICBitmapDitherTypeNone, | 
					
						
							|  |  |  |  | 					nullptr, | 
					
						
							|  |  |  |  | 					0.f, | 
					
						
							|  |  |  |  | 					WICBitmapPaletteTypeMedianCut | 
					
						
							|  |  |  |  | 				); | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			if (SUCCEEDED(hr)) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				// <20><> WIC λͼ<CEBB><CDBC><EFBFBD><EFBFBD>һ<EFBFBD><D2BB> Direct2D λͼ
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 				hr = d2d.render_target->CreateBitmapFromWicBitmap( | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 					converter, | 
					
						
							|  |  |  |  | 					nullptr, | 
					
						
							|  |  |  |  | 					bitmap | 
					
						
							|  |  |  |  | 				); | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (SUCCEEDED(hr)) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				bitmap_cache_.insert(std::make_pair(hash_code, *bitmap)); | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			// <20>ͷ<EFBFBD><CDB7><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Դ
 | 
					
						
							|  |  |  |  | 			SafeRelease(decoder); | 
					
						
							|  |  |  |  | 			SafeRelease(source); | 
					
						
							|  |  |  |  | 			SafeRelease(stream); | 
					
						
							|  |  |  |  | 			SafeRelease(converter); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			return hr; | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 		HRESULT GraphicsDevice::CreateBitmapFromResource(Resource & res, ID2D1Bitmap ** bitmap) | 
					
						
							|  |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (d2d.imaging_factory == nullptr || | 
					
						
							|  |  |  |  | 				d2d.render_target == nullptr) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				return E_UNEXPECTED; | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			if (bitmap == nullptr) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				return E_POINTER; | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			size_t hash_code = res.GetHashCode(); | 
					
						
							|  |  |  |  | 			if (bitmap_cache_.find(hash_code) != bitmap_cache_.end()) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				*bitmap = bitmap_cache_[hash_code]; | 
					
						
							|  |  |  |  | 				return S_OK; | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			HRESULT hr; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			HINSTANCE				hinstance = GetModuleHandle(nullptr); | 
					
						
							|  |  |  |  | 			IWICBitmapDecoder*		decoder = nullptr; | 
					
						
							|  |  |  |  | 			IWICBitmapFrameDecode*	source = nullptr; | 
					
						
							|  |  |  |  | 			IWICStream*				stream = nullptr; | 
					
						
							|  |  |  |  | 			IWICFormatConverter*	converter = nullptr; | 
					
						
							| 
									
										
										
										
											2018-11-12 23:32:50 +08:00
										 |  |  |  | 			ResourceData			buffer; | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			// <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Դ
 | 
					
						
							| 
									
										
										
										
											2018-11-12 23:32:50 +08:00
										 |  |  |  | 			hr = res.Load(&buffer) ? S_OK : E_FAIL; | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 			if (SUCCEEDED(hr)) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				// <20><><EFBFBD><EFBFBD> WIC <20><>
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 				hr = d2d.imaging_factory->CreateStream(&stream); | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			if (SUCCEEDED(hr)) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				// <20><>ʼ<EFBFBD><CABC><EFBFBD><EFBFBD>
 | 
					
						
							|  |  |  |  | 				hr = stream->InitializeFromMemory( | 
					
						
							| 
									
										
										
										
											2018-11-12 23:32:50 +08:00
										 |  |  |  | 					static_cast<WICInProcPointer>(buffer.buffer), | 
					
						
							|  |  |  |  | 					buffer.buffer_size | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				); | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			if (SUCCEEDED(hr)) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				// <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ľ<EFBFBD><C4BD><EFBFBD><EFBFBD><EFBFBD>
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 				hr = d2d.imaging_factory->CreateDecoderFromStream( | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 					stream, | 
					
						
							|  |  |  |  | 					nullptr, | 
					
						
							|  |  |  |  | 					WICDecodeMetadataCacheOnLoad, | 
					
						
							|  |  |  |  | 					&decoder | 
					
						
							|  |  |  |  | 				); | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			if (SUCCEEDED(hr)) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				// <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ʼ<EFBFBD><CABC><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
 | 
					
						
							|  |  |  |  | 				hr = decoder->GetFrame(0, &source); | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			if (SUCCEEDED(hr)) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				// <20><><EFBFBD><EFBFBD>ͼƬ<CDBC><C6AC>ʽת<CABD><D7AA><EFBFBD><EFBFBD>
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 				hr = d2d.imaging_factory->CreateFormatConverter(&converter); | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			if (SUCCEEDED(hr)) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				// ͼƬ<CDBC><C6AC>ʽת<CABD><D7AA><EFBFBD><EFBFBD> 32bppPBGRA
 | 
					
						
							|  |  |  |  | 				hr = converter->Initialize( | 
					
						
							|  |  |  |  | 					source, | 
					
						
							|  |  |  |  | 					GUID_WICPixelFormat32bppPBGRA, | 
					
						
							|  |  |  |  | 					WICBitmapDitherTypeNone, | 
					
						
							|  |  |  |  | 					nullptr, | 
					
						
							|  |  |  |  | 					0.f, | 
					
						
							|  |  |  |  | 					WICBitmapPaletteTypeMedianCut | 
					
						
							|  |  |  |  | 				); | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			if (SUCCEEDED(hr)) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				// <20><> WIC λͼ<CEBB><CDBC><EFBFBD><EFBFBD>һ<EFBFBD><D2BB> Direct2D λͼ
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 				hr = d2d.render_target->CreateBitmapFromWicBitmap( | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 					converter, | 
					
						
							|  |  |  |  | 					nullptr, | 
					
						
							|  |  |  |  | 					bitmap | 
					
						
							|  |  |  |  | 				); | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (SUCCEEDED(hr)) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				bitmap_cache_.insert(std::make_pair(hash_code, *bitmap)); | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			// <20>ͷ<EFBFBD><CDB7><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Դ
 | 
					
						
							|  |  |  |  | 			SafeRelease(decoder); | 
					
						
							|  |  |  |  | 			SafeRelease(source); | 
					
						
							|  |  |  |  | 			SafeRelease(stream); | 
					
						
							|  |  |  |  | 			SafeRelease(converter); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			return hr; | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 		HRESULT GraphicsDevice::Resize(UINT32 width, UINT32 height) | 
					
						
							|  |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (!d2d.render_target) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				return E_UNEXPECTED; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			d2d.render_target->Resize(D2D1::SizeU(width, height)); | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			return S_OK; | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 		HRESULT GraphicsDevice::SetTransform(const math::Matrix & matrix) | 
					
						
							|  |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (!d2d.render_target) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				return E_UNEXPECTED; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			d2d.render_target->SetTransform(matrix); | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			return S_OK; | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 		HRESULT GraphicsDevice::SetBrushOpacity(float opacity) | 
					
						
							|  |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (!d2d.render_target) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				return E_UNEXPECTED; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			d2d.solid_brush->SetOpacity(opacity); | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			return S_OK; | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 		HRESULT GraphicsDevice::SetTextStyle( | 
					
						
							|  |  |  |  | 			const Color & color, | 
					
						
							|  |  |  |  | 			bool has_outline, | 
					
						
							|  |  |  |  | 			const Color & outline_color, | 
					
						
							|  |  |  |  | 			float outline_width, | 
					
						
							|  |  |  |  | 			StrokeStyle outline_stroke | 
					
						
							|  |  |  |  | 		) | 
					
						
							|  |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (!d2d.text_renderer) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				return E_UNEXPECTED; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			d2d.text_renderer->SetTextStyle( | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 				color, | 
					
						
							|  |  |  |  | 				has_outline, | 
					
						
							|  |  |  |  | 				outline_color, | 
					
						
							|  |  |  |  | 				outline_width, | 
					
						
							|  |  |  |  | 				static_cast<D2D1_LINE_JOIN>(outline_stroke) | 
					
						
							|  |  |  |  | 			); | 
					
						
							|  |  |  |  | 			return S_OK; | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 		void GraphicsDevice::SetBackgroundColor(const Color& color) | 
					
						
							|  |  |  |  | 		{ | 
					
						
							|  |  |  |  | 			clear_color_ = color; | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 		void GraphicsDevice::DrawDebugInfo() | 
					
						
							|  |  |  |  | 		{ | 
					
						
							|  |  |  |  | 			static int render_times_ = 0; | 
					
						
							| 
									
										
										
										
											2018-11-11 16:32:12 +08:00
										 |  |  |  | 			static auto last_render_time_ = time::Now(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			int64_t duration = (time::Now() - last_render_time_).Milliseconds(); | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 			if (!fps_text_format_) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				ThrowIfFailed( | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 					d2d.write_factory->CreateTextFormat( | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 						L"", | 
					
						
							|  |  |  |  | 						nullptr, | 
					
						
							|  |  |  |  | 						DWRITE_FONT_WEIGHT_NORMAL, | 
					
						
							|  |  |  |  | 						DWRITE_FONT_STYLE_NORMAL, | 
					
						
							|  |  |  |  | 						DWRITE_FONT_STRETCH_NORMAL, | 
					
						
							|  |  |  |  | 						20, | 
					
						
							|  |  |  |  | 						L"", | 
					
						
							|  |  |  |  | 						&fps_text_format_ | 
					
						
							|  |  |  |  | 					) | 
					
						
							|  |  |  |  | 				); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 				ThrowIfFailed( | 
					
						
							|  |  |  |  | 					fps_text_format_->SetWordWrapping( | 
					
						
							|  |  |  |  | 						DWRITE_WORD_WRAPPING_NO_WRAP | 
					
						
							|  |  |  |  | 					) | 
					
						
							|  |  |  |  | 				); | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			++render_times_; | 
					
						
							| 
									
										
										
										
											2018-11-11 16:32:12 +08:00
										 |  |  |  | 			if (duration >= 100LL) | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				wchar_t fps_text[12] = {}; | 
					
						
							| 
									
										
										
										
											2018-11-11 16:32:12 +08:00
										 |  |  |  | 				int len = swprintf_s(fps_text, L"FPS: %.1f", 1000.f / static_cast<float>(duration) * render_times_); | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 				last_render_time_ = time::Now(); | 
					
						
							|  |  |  |  | 				render_times_ = 0; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 				SafeRelease(fps_text_layout_); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 				ThrowIfFailed( | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 					d2d.write_factory->CreateTextLayout( | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 						fps_text, | 
					
						
							|  |  |  |  | 						static_cast<UINT32>(len), | 
					
						
							|  |  |  |  | 						fps_text_format_, | 
					
						
							|  |  |  |  | 						0, | 
					
						
							|  |  |  |  | 						0, | 
					
						
							|  |  |  |  | 						&fps_text_layout_ | 
					
						
							|  |  |  |  | 					) | 
					
						
							|  |  |  |  | 				); | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 			if (fps_text_layout_) | 
					
						
							|  |  |  |  | 			{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 				d2d.render_target->SetTransform(D2D1::Matrix3x2F::Identity()); | 
					
						
							|  |  |  |  | 				d2d.solid_brush->SetOpacity(1.0f); | 
					
						
							|  |  |  |  | 				d2d.text_renderer->SetTextStyle( | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 					D2D1::ColorF(D2D1::ColorF::White), | 
					
						
							|  |  |  |  | 					TRUE, | 
					
						
							|  |  |  |  | 					D2D1::ColorF(D2D1::ColorF::Black, 0.4f), | 
					
						
							|  |  |  |  | 					1.5f, | 
					
						
							|  |  |  |  | 					D2D1_LINE_JOIN_ROUND | 
					
						
							|  |  |  |  | 				); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 				fps_text_layout_->Draw(nullptr, d2d.text_renderer, 10, 0); | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			} | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 		void GraphicsDevice::CreateDeviceResources(HWND hwnd) | 
					
						
							|  |  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (!d2d.render_target) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				RECT rc; | 
					
						
							|  |  |  |  | 				::GetClientRect(hwnd, &rc); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 				D2D1_SIZE_U size = D2D1::SizeU( | 
					
						
							|  |  |  |  | 					rc.right - rc.left, | 
					
						
							|  |  |  |  | 					rc.bottom - rc.top | 
					
						
							|  |  |  |  | 				); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 				// <20><><EFBFBD><EFBFBD><EFBFBD>豸<EFBFBD><E8B1B8><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Դ<EFBFBD><D4B4><EFBFBD><EFBFBD>Щ<EFBFBD><D0A9>ԴӦ<D4B4><D3A6> Direct2D <20>豸<EFBFBD><E8B1B8>ʧʱ<CAA7>ؽ<EFBFBD>
 | 
					
						
							|  |  |  |  | 				// <20><><EFBFBD><EFBFBD>һ<EFBFBD><D2BB> Direct2D <20><>ȾĿ<C8BE><C4BF>
 | 
					
						
							|  |  |  |  | 				ThrowIfFailed( | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 					d2d.factory->CreateHwndRenderTarget( | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 						D2D1::RenderTargetProperties(), | 
					
						
							|  |  |  |  | 						D2D1::HwndRenderTargetProperties( | 
					
						
							|  |  |  |  | 							hwnd, | 
					
						
							|  |  |  |  | 							size, | 
					
						
							|  |  |  |  | 							D2D1_PRESENT_OPTIONS_NONE), | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 						&d2d.render_target | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 					) | 
					
						
							|  |  |  |  | 				); | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (!d2d.solid_brush) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				ThrowIfFailed( | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 					d2d.render_target->CreateSolidColorBrush( | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 						D2D1::ColorF(D2D1::ColorF::White), | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 						&d2d.solid_brush | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 					) | 
					
						
							|  |  |  |  | 				); | 
					
						
							|  |  |  |  | 			} | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 			if (!d2d.text_renderer) | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 			{ | 
					
						
							|  |  |  |  | 				ThrowIfFailed( | 
					
						
							|  |  |  |  | 					ITextRenderer::Create( | 
					
						
							| 
									
										
										
										
											2018-11-12 20:46:54 +08:00
										 |  |  |  | 						&d2d.text_renderer, | 
					
						
							|  |  |  |  | 						d2d.factory, | 
					
						
							|  |  |  |  | 						d2d.render_target, | 
					
						
							|  |  |  |  | 						d2d.solid_brush | 
					
						
							| 
									
										
										
										
											2018-11-12 02:10:35 +08:00
										 |  |  |  | 					) | 
					
						
							|  |  |  |  | 				); | 
					
						
							| 
									
										
										
										
											2018-11-08 21:39:26 +08:00
										 |  |  |  | 			} | 
					
						
							|  |  |  |  | 		} | 
					
						
							|  |  |  |  | 	} | 
					
						
							|  |  |  |  | } |