/* Copyright (C) 2013 Birunthan Mohanathas This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "StdAfx.h" #include "../Platform.h" #include "CanvasD2D.h" #include "TextFormatD2D.h" #include "Util/DWriteFontCollectionLoader.h" #include "Util/DWriteHelpers.h" #include "Util/WICBitmapLockGDIP.h" #include "../../Library/Litestep.h" namespace { D2D1_COLOR_F ToColorF(const Gdiplus::Color& color) { return D2D1::ColorF(color.GetR() / 255.0f, color.GetG() / 255.0f, color.GetB() / 255.0f, color.GetA() / 255.0f); } D2D1_RECT_F ToRectF(const Gdiplus::Rect& rect) { return D2D1::RectF((FLOAT)rect.X, (FLOAT)rect.Y, (FLOAT)(rect.X + rect.Width), (FLOAT)(rect.Y + rect.Height)); } D2D1_RECT_F ToRectF(const Gdiplus::RectF& rect) { return D2D1::RectF(rect.X, rect.Y, rect.X + rect.Width, rect.Y + rect.Height); } } // namespace namespace Gfx { UINT CanvasD2D::c_Instances = 0; Microsoft::WRL::ComPtr CanvasD2D::c_D2DFactory; Microsoft::WRL::ComPtr CanvasD2D::c_DWFactory; Microsoft::WRL::ComPtr CanvasD2D::c_DWGDIInterop; Microsoft::WRL::ComPtr CanvasD2D::c_WICFactory; CanvasD2D::CanvasD2D() : Canvas(), m_Bitmap(), m_TextAntiAliasing(false), m_CanUseAxisAlignClip(false) { } CanvasD2D::~CanvasD2D() { Finalize(); } bool CanvasD2D::Initialize() { ++c_Instances; if (c_Instances == 1) { if (!Platform::IsAtLeastWin7()) return false; D2D1_FACTORY_OPTIONS fo = {}; #ifdef _DEBUG fo.debugLevel = D2D1_DEBUG_LEVEL_INFORMATION; #endif HRESULT hr = D2D1CreateFactory( D2D1_FACTORY_TYPE_SINGLE_THREADED, fo, c_D2DFactory.GetAddressOf()); if (FAILED(hr)) return false; hr = CoCreateInstance( CLSID_WICImagingFactory, nullptr, CLSCTX_INPROC_SERVER, IID_IWICImagingFactory, (LPVOID*)c_WICFactory.GetAddressOf()); if (FAILED(hr)) return false; hr = DWriteCreateFactory( DWRITE_FACTORY_TYPE_SHARED, __uuidof(c_DWFactory), (IUnknown**)c_DWFactory.GetAddressOf()); if (FAILED(hr)) return false; hr = c_DWFactory->GetGdiInterop(c_DWGDIInterop.GetAddressOf()); if (FAILED(hr)) return false; hr = c_DWFactory->RegisterFontCollectionLoader(Util::DWriteFontCollectionLoader::GetInstance()); if (FAILED(hr)) return false; } return true; } void CanvasD2D::Finalize() { --c_Instances; if (c_Instances == 0) { c_D2DFactory.Reset(); c_WICFactory.Reset(); c_DWGDIInterop.Reset(); if (c_DWFactory) { c_DWFactory->UnregisterFontCollectionLoader(Util::DWriteFontCollectionLoader::GetInstance()); c_DWFactory.Reset(); } } } void CanvasD2D::Resize(int w, int h) { __super::Resize(w, h); m_Target.Reset(); m_Bitmap.Resize(w, h); m_GdipBitmap.reset(new Gdiplus::Bitmap(w, h, w * 4, PixelFormat32bppPARGB, m_Bitmap.GetData())); m_GdipGraphics.reset(new Gdiplus::Graphics(m_GdipBitmap.get())); } bool CanvasD2D::BeginDraw() { return true; } void CanvasD2D::EndDraw() { EndTargetDraw(); } bool CanvasD2D::BeginTargetDraw() { if (m_Target) return true; const D2D1_PIXEL_FORMAT format = D2D1::PixelFormat( DXGI_FORMAT_B8G8R8A8_UNORM, D2D1_ALPHA_MODE_PREMULTIPLIED); const D2D1_RENDER_TARGET_PROPERTIES properties = D2D1::RenderTargetProperties( D2D1_RENDER_TARGET_TYPE_DEFAULT, format, 0.0f, // Default DPI 0.0f, // Default DPI D2D1_RENDER_TARGET_USAGE_GDI_COMPATIBLE); // A new Direct2D render target must be created for each sequence of Direct2D draw operations // since we use GDI+ to render to the same pixel data. Without creating a new render target // each time, it has been found that Direct2D may overwrite the draws by GDI+ since it is // unaware of the changes made by GDI+. By creating a new render target and then releasing it // before the next GDI+ draw operations, we ensure that the pixel data result is as expected // Once GDI+ drawing is no longer needed, we change to recreate the render target only when the // bitmap size is changed. HRESULT hr = c_D2DFactory->CreateWicBitmapRenderTarget(&m_Bitmap, properties, &m_Target); if (SUCCEEDED(hr)) { SetTextAntiAliasing(m_TextAntiAliasing); m_Target->BeginDraw(); // Apply any transforms that occurred before creation of |m_Target|. UpdateTargetTransform(); return true; } return false; } void CanvasD2D::EndTargetDraw() { if (m_Target) { m_Target->EndDraw(); m_Target.Reset(); } } Gdiplus::Graphics& CanvasD2D::BeginGdiplusContext() { EndTargetDraw(); return *m_GdipGraphics; } void CanvasD2D::EndGdiplusContext() { } HDC CanvasD2D::GetDC() { EndTargetDraw(); HDC dcMemory = CreateCompatibleDC(nullptr); SelectObject(dcMemory, m_Bitmap.GetHandle()); return dcMemory; } void CanvasD2D::ReleaseDC(HDC dc) { DeleteDC(dc); } bool CanvasD2D::IsTransparentPixel(int x, int y) { if (!(x >= 0 && y >= 0 && x < m_W && y < m_H)) return false; bool transparent = true; DWORD* data = (DWORD*)m_Bitmap.GetData(); if (data) { DWORD pixel = data[y * m_W + x]; // Top-down DIB. transparent = (pixel & 0xFF000000) != 0; } return transparent; } void CanvasD2D::UpdateTargetTransform() { Gdiplus::Matrix gdipMatrix; m_GdipGraphics->GetTransform(&gdipMatrix); D2D1_MATRIX_3X2_F d2dMatrix; gdipMatrix.GetElements((Gdiplus::REAL*)&d2dMatrix); m_Target->SetTransform(d2dMatrix); m_CanUseAxisAlignClip = d2dMatrix._12 == 0.0f && d2dMatrix._21 == 0.0f && d2dMatrix._31 == 0.0f && d2dMatrix._32 == 0.0f; } void CanvasD2D::SetTransform(const Gdiplus::Matrix& matrix) { m_GdipGraphics->SetTransform(&matrix); if (m_Target) { UpdateTargetTransform(); } } void CanvasD2D::ResetTransform() { m_GdipGraphics->ResetTransform(); if (m_Target) { m_Target->SetTransform(D2D1::Matrix3x2F::Identity()); } } void CanvasD2D::RotateTransform(float angle, float x, float y, float dx, float dy) { m_GdipGraphics->TranslateTransform(x, y); m_GdipGraphics->RotateTransform(angle); m_GdipGraphics->TranslateTransform(dx, dy); if (m_Target) { UpdateTargetTransform(); } } void CanvasD2D::SetAntiAliasing(bool enable) { // TODO: Set m_Target aliasing? m_GdipGraphics->SetSmoothingMode( enable ? Gdiplus::SmoothingModeHighQuality : Gdiplus::SmoothingModeNone); m_GdipGraphics->SetPixelOffsetMode( enable ? Gdiplus::PixelOffsetModeHighQuality : Gdiplus::PixelOffsetModeDefault); } void CanvasD2D::SetTextAntiAliasing(bool enable) { // TODO: Add support for D2D1_TEXT_ANTIALIAS_MODE_CLEARTYPE? m_TextAntiAliasing = enable; if (m_Target) { m_Target->SetTextAntialiasMode( m_TextAntiAliasing ? D2D1_TEXT_ANTIALIAS_MODE_GRAYSCALE : D2D1_TEXT_ANTIALIAS_MODE_ALIASED); } } void CanvasD2D::Clear(const Gdiplus::Color& color) { if (!m_Target) // Use GDI+ if D2D render target has not been created. { m_GdipGraphics->Clear(color); return; } m_Target->Clear(ToColorF(color)); } void CanvasD2D::DrawTextW(const WCHAR* str, UINT strLen, const TextFormat& format, Gdiplus::RectF& rect, const Gdiplus::SolidBrush& brush) { if (!BeginTargetDraw()) return; Gdiplus::Color color; brush.GetColor(&color); Microsoft::WRL::ComPtr solidBrush; HRESULT hr = m_Target->CreateSolidColorBrush(ToColorF(color), solidBrush.GetAddressOf()); if (FAILED(hr)) return; TextFormatD2D& formatD2D = (TextFormatD2D&)format; if (!formatD2D.CreateLayout( str, strLen, rect.Width, rect.Height, !m_AccurateText && m_TextAntiAliasing)) return; D2D1_POINT_2F drawPosition; drawPosition.x = [&]() { if (!m_AccurateText) { const float xOffset = formatD2D.m_TextFormat->GetFontSize() / 6.0f; switch (formatD2D.GetHorizontalAlignment()) { case HorizontalAlignment::Left: return rect.X + xOffset; case HorizontalAlignment::Right: return rect.X - xOffset; } } return rect.X; } (); drawPosition.y = [&]() { // GDI+ compatibility. float yPos = rect.Y - formatD2D.m_LineGap; switch (formatD2D.GetVerticalAlignment()) { case VerticalAlignment::Bottom: yPos -= formatD2D.m_ExtraHeight; break; case VerticalAlignment::Center: yPos -= formatD2D.m_ExtraHeight / 2; break; } return yPos; } (); if (formatD2D.m_Trimming) { D2D1_RECT_F clipRect = ToRectF(rect); if (m_CanUseAxisAlignClip) { m_Target->PushAxisAlignedClip(clipRect, D2D1_ANTIALIAS_MODE_ALIASED); } else { const D2D1_LAYER_PARAMETERS layerParams = D2D1::LayerParameters(clipRect, nullptr, D2D1_ANTIALIAS_MODE_ALIASED); m_Target->PushLayer(layerParams, nullptr); } } m_Target->DrawTextLayout(drawPosition, formatD2D.m_TextLayout.Get(), solidBrush.Get()); if (formatD2D.m_Trimming) { if (m_CanUseAxisAlignClip) { m_Target->PopAxisAlignedClip(); } else { m_Target->PopLayer(); } } } bool CanvasD2D::MeasureTextW(const WCHAR* str, UINT strLen, const TextFormat& format, Gdiplus::RectF& rect) { TextFormatD2D& formatD2D = (TextFormatD2D&)format; const DWRITE_TEXT_METRICS metrics = formatD2D.GetMetrics(str, strLen, !m_AccurateText); rect.Width = metrics.width; rect.Height = metrics.height; return true; } bool CanvasD2D::MeasureTextLinesW(const WCHAR* str, UINT strLen, const TextFormat& format, Gdiplus::RectF& rect, UINT& lines) { TextFormatD2D& formatD2D = (TextFormatD2D&)format; formatD2D.m_TextFormat->SetWordWrapping(DWRITE_WORD_WRAPPING_WRAP); const DWRITE_TEXT_METRICS metrics = formatD2D.GetMetrics(str, strLen, !m_AccurateText, rect.Width); rect.Width = metrics.width; rect.Height = metrics.height; lines = metrics.lineCount; if (rect.Height > 0.0f) { // GDI+ draws multi-line text even though the last line may be clipped slightly at the // bottom. This is a workaround to emulate that behaviour. rect.Height += 1.0f; } else { // GDI+ compatibility: Zero height text has no visible lines. lines = 0; } return true; } void CanvasD2D::DrawBitmap(Gdiplus::Bitmap* bitmap, const Gdiplus::Rect& dstRect, const Gdiplus::Rect& srcRect) { if (srcRect.Width != dstRect.Width || srcRect.Height != dstRect.Height) { // If the bitmap needs to be scaled, get rid of the D2D target and use the GDI+ code path // to draw the bitmap. This is due to antialiasing differences between GDI+ and D2D on // scaled bitmaps. EndTargetDraw(); } if (!m_Target) // Use GDI+ if D2D render target has not been created. { m_GdipGraphics->DrawImage( bitmap, dstRect, srcRect.X, srcRect.Y, srcRect.Width, srcRect.Height, Gdiplus::UnitPixel); return; } // The D2D DrawBitmap seems to perform exactly like Gdiplus::Graphics::DrawImage since we are // not using a hardware accelerated render target. Nevertheless, we will use it to avoid // the EndDraw() call needed for GDI+ drawing. Util::WICBitmapLockGDIP* bitmapLock = new Util::WICBitmapLockGDIP(); Gdiplus::Rect lockRect(0, 0, bitmap->GetWidth(), bitmap->GetHeight()); Gdiplus::Status status = bitmap->LockBits( &lockRect, Gdiplus::ImageLockModeRead, PixelFormat32bppPARGB, bitmapLock->GetBitmapData()); if (status == Gdiplus::Ok) { D2D1_BITMAP_PROPERTIES props = D2D1::BitmapProperties( D2D1::PixelFormat(DXGI_FORMAT_B8G8R8A8_UNORM, D2D1_ALPHA_MODE_PREMULTIPLIED)); Microsoft::WRL::ComPtr d2dBitmap; HRESULT hr = m_Target->CreateSharedBitmap( __uuidof(IWICBitmapLock), bitmapLock, &props, d2dBitmap.GetAddressOf()); if (SUCCEEDED(hr)) { auto rDst = ToRectF(dstRect); auto rSrc = ToRectF(srcRect); m_Target->DrawBitmap(d2dBitmap.Get(), rDst, 1.0F, D2D1_BITMAP_INTERPOLATION_MODE_LINEAR, rSrc); } // D2D will still use the pixel data after this call (at the next Flush() or EndDraw()). bitmap->UnlockBits(bitmapLock->GetBitmapData()); } bitmapLock->Release(); } void CanvasD2D::FillRectangle(Gdiplus::Rect& rect, const Gdiplus::SolidBrush& brush) { if (!m_Target) // Use GDI+ if D2D render target has not been created. { m_GdipGraphics->FillRectangle(&brush, rect); return; } Gdiplus::Color color; brush.GetColor(&color); Microsoft::WRL::ComPtr solidBrush; HRESULT hr = m_Target->CreateSolidColorBrush(ToColorF(color), solidBrush.GetAddressOf()); if (SUCCEEDED(hr)) { m_Target->FillRectangle(ToRectF(rect), solidBrush.Get()); } } } // namespace Gfx