Magic_Game/src/kiwano/renderer/Renderer.cpp

533 lines
9.7 KiB
C++
Raw Normal View History

2019-04-11 14:40:54 +08:00
// Copyright (c) 2016-2018 Kiwano - 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.
2019-08-14 21:52:49 +08:00
#include "Renderer.h"
#include "../base/Logger.h"
#include "../base/Window.h"
2019-08-16 00:50:54 +08:00
#include "../utils/FileUtil.h"
2019-04-11 14:40:54 +08:00
namespace kiwano
{
Renderer::Renderer()
: hwnd_(nullptr)
, vsync_(true)
, clear_color_(Color::Black)
{
}
Renderer::~Renderer()
{
}
2019-08-12 14:51:54 +08:00
void Renderer::SetupComponent()
{
2019-04-11 14:40:54 +08:00
KGE_LOG(L"Creating device resources");
hwnd_ = Window::GetInstance()->GetHandle();
2019-04-22 13:21:12 +08:00
ThrowIfFailed(hwnd_ ? S_OK : E_FAIL);
2019-07-30 00:41:06 +08:00
d2d_res_ = nullptr;
d3d_res_ = nullptr;
2019-04-22 13:21:12 +08:00
drawing_state_block_ = nullptr;
ThrowIfFailed(
2019-07-30 00:41:06 +08:00
ID2DDeviceResources::Create(
&d2d_res_
)
);
ThrowIfFailed(
#if defined(KGE_USE_DIRECTX10)
ID3D10DeviceResources::Create(
&d3d_res_,
2019-08-13 21:16:38 +08:00
d2d_res_.get(),
2019-07-30 00:41:06 +08:00
hwnd_
)
#else
ID3D11DeviceResources::Create(
&d3d_res_,
2019-08-13 21:16:38 +08:00
d2d_res_.get(),
hwnd_
2019-04-22 13:21:12 +08:00
)
2019-07-30 00:41:06 +08:00
#endif
2019-04-22 13:21:12 +08:00
);
2019-04-22 13:21:12 +08:00
ThrowIfFailed(
2019-07-30 00:41:06 +08:00
d2d_res_->GetFactory()->CreateDrawingStateBlock(
&drawing_state_block_
2019-04-22 13:21:12 +08:00
)
);
2019-04-22 13:21:12 +08:00
ThrowIfFailed(
CreateDeviceResources()
);
output_size_ = Window::GetInstance()->GetSize();
}
void Renderer::DestroyComponent()
{
2019-04-11 14:40:54 +08:00
KGE_LOG(L"Destroying device resources");
2019-08-13 21:16:38 +08:00
drawing_state_block_.reset();
solid_color_brush_.reset();
d2d_res_.reset();
d3d_res_.reset();
}
2019-07-30 13:32:10 +08:00
void Renderer::BeforeRender()
{
2019-08-16 00:50:54 +08:00
HRESULT hr = S_OK;
if (!IsValid())
{
hr = E_UNEXPECTED;
}
if (SUCCEEDED(hr))
{
render_target_->SaveDrawingState(drawing_state_block_.get());
BeginDraw();
}
if (SUCCEEDED(hr))
{
hr = d3d_res_->ClearRenderTarget(clear_color_);
}
ThrowIfFailed(hr);
2019-07-30 13:32:10 +08:00
}
void Renderer::AfterRender()
{
2019-08-16 00:50:54 +08:00
HRESULT hr = S_OK;
if (!IsValid())
{
hr = E_UNEXPECTED;
}
if (SUCCEEDED(hr))
{
EndDraw();
render_target_->RestoreDrawingState(drawing_state_block_.get());
}
if (SUCCEEDED(hr))
{
hr = d3d_res_->Present(vsync_);
}
if (hr == DXGI_ERROR_DEVICE_REMOVED || hr == DXGI_ERROR_DEVICE_RESET)
{
// <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>Դ
hr = HandleDeviceLost();
}
ThrowIfFailed(hr);
2019-07-30 13:32:10 +08:00
}
void Renderer::HandleMessage(HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam)
{
switch (msg)
{
case WM_SIZE:
{
UINT width = LOWORD(lparam);
UINT height = HIWORD(lparam);
Resize(width, height);
break;
}
}
}
HRESULT Renderer::CreateDeviceResources()
{
2019-08-16 00:50:54 +08:00
HRESULT hr = InitDeviceResources(
d2d_res_->GetDeviceContext(),
d2d_res_
);
if (SUCCEEDED(hr))
{
SetAntialiasMode(antialias_);
SetTextAntialiasMode(text_antialias_);
}
return hr;
}
HRESULT Renderer::HandleDeviceLost()
{
2019-07-30 00:41:06 +08:00
HRESULT hr = d3d_res_->HandleDeviceLost();
if (SUCCEEDED(hr))
{
hr = CreateDeviceResources();
}
return hr;
}
2019-08-16 00:50:54 +08:00
void Renderer::CreateImage(Image& image, Resource const& res)
{
2019-08-16 00:50:54 +08:00
HRESULT hr = S_OK;
if (!d2d_res_)
{
2019-08-16 00:50:54 +08:00
hr = E_UNEXPECTED;
}
2019-08-16 00:50:54 +08:00
ComPtr<ID2D1Bitmap> bitmap;
if (res.IsFileType())
{
2019-08-16 00:50:54 +08:00
hr = d2d_res_->CreateBitmapFromFile(bitmap, res.GetFileName());
2019-07-29 09:40:39 +08:00
}
2019-08-16 00:50:54 +08:00
else
{
2019-08-16 00:50:54 +08:00
hr = d2d_res_->CreateBitmapFromResource(bitmap, res);
}
2019-08-16 00:50:54 +08:00
if (SUCCEEDED(hr))
{
2019-08-16 00:50:54 +08:00
image.SetBitmap(bitmap);
}
2019-08-16 00:50:54 +08:00
if (FAILED(hr))
2019-08-14 08:52:01 +08:00
{
2019-08-16 00:50:54 +08:00
KGE_WARNING_LOG(L"Load image failed with HRESULT of %08X!", hr);
2019-08-14 08:52:01 +08:00
}
}
2019-08-16 00:50:54 +08:00
void Renderer::CreateGifImage(GifImage& image, Resource const& res)
2019-08-14 08:52:01 +08:00
{
HRESULT hr = S_OK;
2019-08-16 00:50:54 +08:00
if (!d2d_res_)
2019-08-14 08:52:01 +08:00
{
hr = E_UNEXPECTED;
}
2019-08-16 00:50:54 +08:00
ComPtr<IWICBitmapDecoder> decoder;
if (res.IsFileType())
2019-08-14 08:52:01 +08:00
{
2019-08-16 00:50:54 +08:00
if (!FileUtil::ExistsFile(res.GetFileName().c_str()))
{
KGE_WARNING_LOG(L"Gif image file '%s' not found!", res.GetFileName().c_str());
hr = E_FAIL;
}
if (SUCCEEDED(hr))
{
hr = d2d_res_->GetWICImagingFactory()->CreateDecoderFromFilename(
res.GetFileName().c_str(),
nullptr,
GENERIC_READ,
WICDecodeMetadataCacheOnLoad,
&decoder
);
}
}
else
{
LPVOID buffer;
DWORD buffer_size;
HRESULT hr = res.Load(buffer, buffer_size) ? S_OK : E_FAIL;
ComPtr<IWICStream> stream;
if (SUCCEEDED(hr))
{
hr = d2d_res_->GetWICImagingFactory()->CreateStream(&stream);
}
if (SUCCEEDED(hr))
{
hr = stream->InitializeFromMemory(
static_cast<WICInProcPointer>(buffer),
buffer_size
);
}
if (SUCCEEDED(hr))
{
hr = d2d_res_->GetWICImagingFactory()->CreateDecoderFromStream(
stream.get(),
nullptr,
WICDecodeMetadataCacheOnLoad,
&decoder
);
}
2019-08-14 08:52:01 +08:00
}
2019-08-14 08:52:01 +08:00
if (SUCCEEDED(hr))
{
2019-08-16 00:50:54 +08:00
image.SetDecoder(decoder);
2019-08-14 08:52:01 +08:00
}
2019-08-16 00:50:54 +08:00
if (FAILED(hr))
{
KGE_WARNING_LOG(L"Load GIF image failed with HRESULT of %08X!", hr);
}
}
2019-08-15 11:22:51 +08:00
void Renderer::CreateTextFormat(TextFormat& format, Font const& font)
{
HRESULT hr = S_OK;
if (!d2d_res_)
{
hr = E_UNEXPECTED;
}
ComPtr<IDWriteTextFormat> output;
if (SUCCEEDED(hr))
{
hr = d2d_res_->CreateTextFormat(output, font);
}
if (SUCCEEDED(hr))
{
format.SetTextFormat(output);
}
ThrowIfFailed(hr);
}
void Renderer::CreateTextLayout(TextLayout& layout, String const& text, TextStyle const& style, TextFormat const& format)
{
HRESULT hr = S_OK;
if (!d2d_res_)
{
hr = E_UNEXPECTED;
}
ComPtr<IDWriteTextLayout> output;
if (SUCCEEDED(hr))
{
hr = d2d_res_->CreateTextLayout(
output,
text,
style,
format.GetTextFormat()
);
}
if (SUCCEEDED(hr))
{
layout.SetTextLayout(output);
}
ThrowIfFailed(hr);
}
void Renderer::CreateLineGeometry(Geometry& geo, Point const& begin_pos, Point const& end_pos)
{
HRESULT hr = S_OK;
2019-08-16 00:50:54 +08:00
if (!d2d_res_)
{
hr = E_UNEXPECTED;
}
ComPtr<ID2D1PathGeometry> path_geo;
ComPtr<ID2D1GeometrySink> path_sink;
if (SUCCEEDED(hr))
{
hr = d2d_res_->GetFactory()->CreatePathGeometry(&path_geo);
}
if (SUCCEEDED(hr))
{
hr = path_geo->Open(&path_sink);
}
if (SUCCEEDED(hr))
{
path_sink->BeginFigure(DX::ConvertToPoint2F(begin_pos), D2D1_FIGURE_BEGIN_FILLED);
path_sink->AddLine(DX::ConvertToPoint2F(end_pos));
path_sink->EndFigure(D2D1_FIGURE_END_OPEN);
hr = path_sink->Close();
}
if (SUCCEEDED(hr))
{
geo.SetGeometry(path_geo);
}
ThrowIfFailed(hr);
}
void Renderer::CreateRectGeometry(Geometry& geo, Rect const& rect)
{
HRESULT hr = S_OK;
2019-08-16 00:50:54 +08:00
if (!d2d_res_)
{
hr = E_UNEXPECTED;
}
ComPtr<ID2D1RectangleGeometry> output;
if (SUCCEEDED(hr))
{
hr = d2d_res_->GetFactory()->CreateRectangleGeometry(DX::ConvertToRectF(rect), &output);
}
if (SUCCEEDED(hr))
{
geo.SetGeometry(output);
}
ThrowIfFailed(hr);
}
void Renderer::CreateRoundedRectGeometry(Geometry& geo, Rect const& rect, Vec2 const& radius)
{
HRESULT hr = S_OK;
2019-08-16 00:50:54 +08:00
if (!d2d_res_)
{
hr = E_UNEXPECTED;
}
ComPtr<ID2D1RoundedRectangleGeometry> output;
if (SUCCEEDED(hr))
{
hr = d2d_res_->GetFactory()->CreateRoundedRectangleGeometry(
D2D1::RoundedRect(
DX::ConvertToRectF(rect),
radius.x,
radius.y
),
&output);
}
if (SUCCEEDED(hr))
{
geo.SetGeometry(output);
}
ThrowIfFailed(hr);
}
void Renderer::CreateEllipseGeometry(Geometry& geo, Point const& center, Vec2 const& radius)
{
HRESULT hr = S_OK;
2019-08-16 00:50:54 +08:00
if (!d2d_res_)
{
hr = E_UNEXPECTED;
}
ComPtr<ID2D1EllipseGeometry> output;
if (SUCCEEDED(hr))
{
hr = d2d_res_->GetFactory()->CreateEllipseGeometry(
D2D1::Ellipse(
DX::ConvertToPoint2F(center),
radius.x,
radius.y
),
&output);
}
if (SUCCEEDED(hr))
{
geo.SetGeometry(output);
}
ThrowIfFailed(hr);
}
void Renderer::CreatePathGeometrySink(GeometrySink& sink)
{
HRESULT hr = S_OK;
2019-08-16 00:50:54 +08:00
if (!d2d_res_)
{
hr = E_UNEXPECTED;
}
ComPtr<ID2D1PathGeometry> output;
if (SUCCEEDED(hr))
{
hr = d2d_res_->GetFactory()->CreatePathGeometry(&output);
}
if (SUCCEEDED(hr))
{
sink.SetPathGeometry(output);
}
ThrowIfFailed(hr);
}
2019-08-16 00:50:54 +08:00
void Renderer::CreateImageRenderTarget(ImageRenderTarget& render_target)
{
2019-08-14 08:52:01 +08:00
HRESULT hr = S_OK;
2019-08-16 00:50:54 +08:00
if (!d2d_res_)
2019-08-14 08:52:01 +08:00
{
hr = E_UNEXPECTED;
}
2019-08-16 00:50:54 +08:00
ComPtr<ID2D1BitmapRenderTarget> output;
2019-08-14 08:52:01 +08:00
if (SUCCEEDED(hr))
{
2019-08-16 00:50:54 +08:00
hr = d2d_res_->GetDeviceContext()->CreateCompatibleRenderTarget(&output);
2019-08-15 11:22:51 +08:00
}
if (SUCCEEDED(hr))
{
2019-08-16 00:50:54 +08:00
hr = render_target.InitDeviceResources(output, d2d_res_);
2019-08-14 08:52:01 +08:00
}
ThrowIfFailed(hr);
}
void Renderer::SetVSyncEnabled(bool enabled)
{
vsync_ = enabled;
}
2019-08-14 08:52:01 +08:00
void Renderer::Resize(UINT width, UINT height)
{
2019-08-14 08:52:01 +08:00
HRESULT hr = S_OK;
if (!d3d_res_)
{
hr = E_UNEXPECTED;
}
if (SUCCEEDED(hr))
{
2019-08-14 08:52:01 +08:00
output_size_.x = static_cast<float>(width);
output_size_.y = static_cast<float>(height);
hr = d3d_res_->SetLogicalSize(output_size_);
}
2019-08-14 08:52:01 +08:00
ThrowIfFailed(hr);
}
2019-08-16 00:50:54 +08:00
void Renderer::SetClearColor(const Color& color)
{
clear_color_ = color;
}
2019-08-14 23:36:29 +08:00
bool Renderer::CheckVisibility(Size const& content_size, Matrix3x2 const& transform)
2019-08-14 00:28:25 +08:00
{
return Rect{ Point{}, output_size_ }.Intersects(
transform.Transform(Rect{ Point{}, content_size })
);
}
}