2018-04-21 23:09:59 +08:00
|
|
|
#include "..\e2dcustom.h"
|
2018-04-22 00:33:20 +08:00
|
|
|
#include "..\e2dbase.h"
|
2018-04-21 23:09:59 +08:00
|
|
|
|
|
|
|
|
using namespace e2d;
|
|
|
|
|
|
|
|
|
|
CustomTextRenderer::CustomTextRenderer(
|
|
|
|
|
ID2D1Factory* pD2DFactory,
|
|
|
|
|
ID2D1HwndRenderTarget* pRT,
|
|
|
|
|
ID2D1SolidColorBrush* pBrush
|
|
|
|
|
)
|
|
|
|
|
: cRefCount_(0)
|
|
|
|
|
, pD2DFactory_(pD2DFactory)
|
|
|
|
|
, pRT_(pRT)
|
|
|
|
|
, pBrush_(pBrush)
|
|
|
|
|
, sFillColor_()
|
|
|
|
|
, sOutlineColor_()
|
2018-04-22 00:33:20 +08:00
|
|
|
, fOutlineWidth(1)
|
|
|
|
|
, bShowOutline_(TRUE)
|
2018-04-23 17:59:46 +08:00
|
|
|
, nOutlineJoin_(D2D1_LINE_JOIN_MITER)
|
2018-04-21 23:09:59 +08:00
|
|
|
{
|
|
|
|
|
pD2DFactory_->AddRef();
|
|
|
|
|
pRT_->AddRef();
|
|
|
|
|
pBrush_->AddRef();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CustomTextRenderer::~CustomTextRenderer()
|
|
|
|
|
{
|
2018-05-10 20:23:32 +08:00
|
|
|
SafeReleaseInterface(pD2DFactory_);
|
|
|
|
|
SafeReleaseInterface(pRT_);
|
|
|
|
|
SafeReleaseInterface(pBrush_);
|
2018-04-21 23:09:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
STDMETHODIMP_(void) CustomTextRenderer::SetTextStyle(
|
|
|
|
|
CONST D2D1_COLOR_F &fillColor,
|
2018-04-22 14:08:29 +08:00
|
|
|
BOOL hasOutline,
|
2018-04-21 23:09:59 +08:00
|
|
|
CONST D2D1_COLOR_F &outlineColor,
|
2018-04-22 00:33:20 +08:00
|
|
|
FLOAT outlineWidth,
|
|
|
|
|
D2D1_LINE_JOIN outlineJoin
|
2018-04-21 23:09:59 +08:00
|
|
|
)
|
|
|
|
|
{
|
|
|
|
|
sFillColor_ = fillColor;
|
2018-04-22 14:08:29 +08:00
|
|
|
bShowOutline_ = hasOutline;
|
2018-04-21 23:09:59 +08:00
|
|
|
sOutlineColor_ = outlineColor;
|
2018-04-22 00:33:20 +08:00
|
|
|
fOutlineWidth = 2 * outlineWidth;
|
|
|
|
|
nOutlineJoin_ = outlineJoin;
|
2018-04-21 23:09:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
STDMETHODIMP CustomTextRenderer::DrawGlyphRun(
|
|
|
|
|
__maybenull void* clientDrawingContext,
|
|
|
|
|
FLOAT baselineOriginX,
|
|
|
|
|
FLOAT baselineOriginY,
|
|
|
|
|
DWRITE_MEASURING_MODE measuringMode,
|
|
|
|
|
__in DWRITE_GLYPH_RUN const* glyphRun,
|
|
|
|
|
__in DWRITE_GLYPH_RUN_DESCRIPTION const* glyphRunDescription,
|
|
|
|
|
IUnknown* clientDrawingEffect
|
|
|
|
|
)
|
|
|
|
|
{
|
|
|
|
|
HRESULT hr = S_OK;
|
|
|
|
|
|
2018-05-17 15:22:14 +08:00
|
|
|
ID2D1PathGeometry* pPathGeometry = nullptr;
|
2018-04-21 23:09:59 +08:00
|
|
|
hr = pD2DFactory_->CreatePathGeometry(
|
|
|
|
|
&pPathGeometry
|
|
|
|
|
);
|
|
|
|
|
|
2018-05-17 15:22:14 +08:00
|
|
|
ID2D1GeometrySink* pSink = nullptr;
|
2018-04-21 23:09:59 +08:00
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
|
{
|
|
|
|
|
hr = pPathGeometry->Open(
|
|
|
|
|
&pSink
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
|
{
|
|
|
|
|
hr = glyphRun->fontFace->GetGlyphRunOutline(
|
|
|
|
|
glyphRun->fontEmSize,
|
|
|
|
|
glyphRun->glyphIndices,
|
|
|
|
|
glyphRun->glyphAdvances,
|
|
|
|
|
glyphRun->glyphOffsets,
|
|
|
|
|
glyphRun->glyphCount,
|
|
|
|
|
glyphRun->isSideways,
|
|
|
|
|
glyphRun->bidiLevel % 2,
|
|
|
|
|
pSink
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
|
{
|
|
|
|
|
hr = pSink->Close();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
D2D1::Matrix3x2F const matrix = D2D1::Matrix3x2F(
|
|
|
|
|
1.0f, 0.0f,
|
|
|
|
|
0.0f, 1.0f,
|
|
|
|
|
baselineOriginX, baselineOriginY
|
|
|
|
|
);
|
|
|
|
|
|
2018-05-17 15:22:14 +08:00
|
|
|
ID2D1TransformedGeometry* pTransformedGeometry = nullptr;
|
2018-04-21 23:09:59 +08:00
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
|
{
|
|
|
|
|
hr = pD2DFactory_->CreateTransformedGeometry(
|
|
|
|
|
pPathGeometry,
|
|
|
|
|
&matrix,
|
|
|
|
|
&pTransformedGeometry
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-17 15:22:14 +08:00
|
|
|
if (SUCCEEDED(hr) && bShowOutline_)
|
2018-04-21 23:09:59 +08:00
|
|
|
{
|
2018-05-17 15:22:14 +08:00
|
|
|
ID2D1StrokeStyle * pStrokeStyle = nullptr;
|
2018-04-22 00:33:20 +08:00
|
|
|
hr = Renderer::getID2D1Factory()->CreateStrokeStyle(
|
|
|
|
|
D2D1::StrokeStyleProperties(
|
|
|
|
|
D2D1_CAP_STYLE_FLAT,
|
|
|
|
|
D2D1_CAP_STYLE_FLAT,
|
|
|
|
|
D2D1_CAP_STYLE_FLAT,
|
|
|
|
|
nOutlineJoin_,
|
|
|
|
|
2.0f,
|
|
|
|
|
D2D1_DASH_STYLE_SOLID,
|
|
|
|
|
0.0f),
|
2018-05-17 15:22:14 +08:00
|
|
|
nullptr,
|
2018-04-22 00:33:20 +08:00
|
|
|
0,
|
|
|
|
|
&pStrokeStyle
|
2018-04-21 23:09:59 +08:00
|
|
|
);
|
2018-04-22 00:33:20 +08:00
|
|
|
|
2018-05-17 15:22:14 +08:00
|
|
|
if (SUCCEEDED(hr))
|
2018-04-22 00:33:20 +08:00
|
|
|
{
|
|
|
|
|
pBrush_->SetColor(sOutlineColor_);
|
|
|
|
|
|
|
|
|
|
pRT_->DrawGeometry(
|
|
|
|
|
pTransformedGeometry,
|
|
|
|
|
pBrush_,
|
|
|
|
|
fOutlineWidth,
|
|
|
|
|
pStrokeStyle
|
|
|
|
|
);
|
|
|
|
|
}
|
2018-05-17 15:22:14 +08:00
|
|
|
}
|
2018-04-21 23:09:59 +08:00
|
|
|
|
2018-05-17 15:22:14 +08:00
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
|
{
|
2018-04-21 23:09:59 +08:00
|
|
|
pBrush_->SetColor(sFillColor_);
|
|
|
|
|
|
|
|
|
|
pRT_->FillGeometry(
|
|
|
|
|
pTransformedGeometry,
|
|
|
|
|
pBrush_
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-10 20:23:32 +08:00
|
|
|
SafeReleaseInterface(pPathGeometry);
|
|
|
|
|
SafeReleaseInterface(pSink);
|
|
|
|
|
SafeReleaseInterface(pTransformedGeometry);
|
2018-04-21 23:09:59 +08:00
|
|
|
|
|
|
|
|
return hr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
STDMETHODIMP CustomTextRenderer::DrawUnderline(
|
|
|
|
|
__maybenull void* clientDrawingContext,
|
|
|
|
|
FLOAT baselineOriginX,
|
|
|
|
|
FLOAT baselineOriginY,
|
|
|
|
|
__in DWRITE_UNDERLINE const* underline,
|
|
|
|
|
IUnknown* clientDrawingEffect
|
|
|
|
|
)
|
|
|
|
|
{
|
|
|
|
|
HRESULT hr;
|
|
|
|
|
|
|
|
|
|
D2D1_RECT_F rect = D2D1::RectF(
|
|
|
|
|
0,
|
|
|
|
|
underline->offset,
|
|
|
|
|
underline->width,
|
|
|
|
|
underline->offset + underline->thickness
|
|
|
|
|
);
|
|
|
|
|
|
2018-05-17 15:22:14 +08:00
|
|
|
ID2D1RectangleGeometry* pRectangleGeometry = nullptr;
|
2018-04-21 23:09:59 +08:00
|
|
|
hr = pD2DFactory_->CreateRectangleGeometry(
|
|
|
|
|
&rect,
|
|
|
|
|
&pRectangleGeometry
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
D2D1::Matrix3x2F const matrix = D2D1::Matrix3x2F(
|
|
|
|
|
1.0f, 0.0f,
|
|
|
|
|
0.0f, 1.0f,
|
|
|
|
|
baselineOriginX, baselineOriginY
|
|
|
|
|
);
|
|
|
|
|
|
2018-05-17 15:22:14 +08:00
|
|
|
ID2D1TransformedGeometry* pTransformedGeometry = nullptr;
|
2018-04-21 23:09:59 +08:00
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
|
{
|
|
|
|
|
hr = pD2DFactory_->CreateTransformedGeometry(
|
|
|
|
|
pRectangleGeometry,
|
|
|
|
|
&matrix,
|
|
|
|
|
&pTransformedGeometry
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-17 15:22:14 +08:00
|
|
|
if (SUCCEEDED(hr) && bShowOutline_)
|
2018-04-21 23:09:59 +08:00
|
|
|
{
|
2018-05-17 15:22:14 +08:00
|
|
|
ID2D1StrokeStyle * pStrokeStyle = nullptr;
|
2018-04-22 00:33:20 +08:00
|
|
|
hr = Renderer::getID2D1Factory()->CreateStrokeStyle(
|
|
|
|
|
D2D1::StrokeStyleProperties(
|
|
|
|
|
D2D1_CAP_STYLE_FLAT,
|
|
|
|
|
D2D1_CAP_STYLE_FLAT,
|
|
|
|
|
D2D1_CAP_STYLE_FLAT,
|
|
|
|
|
nOutlineJoin_,
|
|
|
|
|
2.0f,
|
|
|
|
|
D2D1_DASH_STYLE_SOLID,
|
|
|
|
|
0.0f),
|
2018-05-17 15:22:14 +08:00
|
|
|
nullptr,
|
2018-04-22 00:33:20 +08:00
|
|
|
0,
|
|
|
|
|
&pStrokeStyle
|
2018-04-21 23:09:59 +08:00
|
|
|
);
|
2018-04-22 00:33:20 +08:00
|
|
|
|
2018-05-17 15:22:14 +08:00
|
|
|
if (SUCCEEDED(hr))
|
2018-04-22 00:33:20 +08:00
|
|
|
{
|
|
|
|
|
pBrush_->SetColor(sOutlineColor_);
|
|
|
|
|
|
|
|
|
|
pRT_->DrawGeometry(
|
|
|
|
|
pTransformedGeometry,
|
|
|
|
|
pBrush_,
|
|
|
|
|
fOutlineWidth,
|
|
|
|
|
pStrokeStyle
|
|
|
|
|
);
|
|
|
|
|
}
|
2018-05-17 15:22:14 +08:00
|
|
|
}
|
2018-04-21 23:09:59 +08:00
|
|
|
|
2018-05-17 15:22:14 +08:00
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
|
{
|
2018-04-21 23:09:59 +08:00
|
|
|
pBrush_->SetColor(sFillColor_);
|
|
|
|
|
|
|
|
|
|
pRT_->FillGeometry(
|
|
|
|
|
pTransformedGeometry,
|
|
|
|
|
pBrush_
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-10 20:23:32 +08:00
|
|
|
SafeReleaseInterface(pRectangleGeometry);
|
|
|
|
|
SafeReleaseInterface(pTransformedGeometry);
|
2018-04-21 23:09:59 +08:00
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
STDMETHODIMP CustomTextRenderer::DrawStrikethrough(
|
|
|
|
|
__maybenull void* clientDrawingContext,
|
|
|
|
|
FLOAT baselineOriginX,
|
|
|
|
|
FLOAT baselineOriginY,
|
|
|
|
|
__in DWRITE_STRIKETHROUGH const* strikethrough,
|
|
|
|
|
IUnknown* clientDrawingEffect
|
|
|
|
|
)
|
|
|
|
|
{
|
|
|
|
|
HRESULT hr;
|
|
|
|
|
|
|
|
|
|
D2D1_RECT_F rect = D2D1::RectF(
|
|
|
|
|
0,
|
|
|
|
|
strikethrough->offset,
|
|
|
|
|
strikethrough->width,
|
|
|
|
|
strikethrough->offset + strikethrough->thickness
|
|
|
|
|
);
|
|
|
|
|
|
2018-05-17 15:22:14 +08:00
|
|
|
ID2D1RectangleGeometry* pRectangleGeometry = nullptr;
|
2018-04-21 23:09:59 +08:00
|
|
|
hr = pD2DFactory_->CreateRectangleGeometry(
|
|
|
|
|
&rect,
|
|
|
|
|
&pRectangleGeometry
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
D2D1::Matrix3x2F const matrix = D2D1::Matrix3x2F(
|
|
|
|
|
1.0f, 0.0f,
|
|
|
|
|
0.0f, 1.0f,
|
|
|
|
|
baselineOriginX, baselineOriginY
|
|
|
|
|
);
|
|
|
|
|
|
2018-05-17 15:22:14 +08:00
|
|
|
ID2D1TransformedGeometry* pTransformedGeometry = nullptr;
|
2018-04-21 23:09:59 +08:00
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
|
{
|
|
|
|
|
hr = pD2DFactory_->CreateTransformedGeometry(
|
|
|
|
|
pRectangleGeometry,
|
|
|
|
|
&matrix,
|
|
|
|
|
&pTransformedGeometry
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-17 15:22:14 +08:00
|
|
|
if (SUCCEEDED(hr) && bShowOutline_)
|
2018-04-21 23:09:59 +08:00
|
|
|
{
|
2018-05-17 15:22:14 +08:00
|
|
|
ID2D1StrokeStyle * pStrokeStyle = nullptr;
|
2018-04-22 00:33:20 +08:00
|
|
|
hr = Renderer::getID2D1Factory()->CreateStrokeStyle(
|
|
|
|
|
D2D1::StrokeStyleProperties(
|
|
|
|
|
D2D1_CAP_STYLE_FLAT,
|
|
|
|
|
D2D1_CAP_STYLE_FLAT,
|
|
|
|
|
D2D1_CAP_STYLE_FLAT,
|
|
|
|
|
nOutlineJoin_,
|
|
|
|
|
2.0f,
|
|
|
|
|
D2D1_DASH_STYLE_SOLID,
|
|
|
|
|
0.0f),
|
2018-05-17 15:22:14 +08:00
|
|
|
nullptr,
|
2018-04-22 00:33:20 +08:00
|
|
|
0,
|
|
|
|
|
&pStrokeStyle
|
2018-04-21 23:09:59 +08:00
|
|
|
);
|
2018-04-22 00:33:20 +08:00
|
|
|
|
2018-05-17 15:22:14 +08:00
|
|
|
if (SUCCEEDED(hr))
|
2018-04-22 00:33:20 +08:00
|
|
|
{
|
|
|
|
|
pBrush_->SetColor(sOutlineColor_);
|
|
|
|
|
|
|
|
|
|
pRT_->DrawGeometry(
|
|
|
|
|
pTransformedGeometry,
|
|
|
|
|
pBrush_,
|
|
|
|
|
fOutlineWidth,
|
|
|
|
|
pStrokeStyle
|
|
|
|
|
);
|
|
|
|
|
}
|
2018-05-17 15:22:14 +08:00
|
|
|
}
|
2018-04-21 23:09:59 +08:00
|
|
|
|
2018-05-17 15:22:14 +08:00
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
|
{
|
2018-04-21 23:09:59 +08:00
|
|
|
pBrush_->SetColor(sFillColor_);
|
|
|
|
|
|
|
|
|
|
pRT_->FillGeometry(
|
|
|
|
|
pTransformedGeometry,
|
|
|
|
|
pBrush_
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-10 20:23:32 +08:00
|
|
|
SafeReleaseInterface(pRectangleGeometry);
|
|
|
|
|
SafeReleaseInterface(pTransformedGeometry);
|
2018-04-21 23:09:59 +08:00
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
STDMETHODIMP CustomTextRenderer::DrawInlineObject(
|
|
|
|
|
__maybenull void* clientDrawingContext,
|
|
|
|
|
FLOAT originX,
|
|
|
|
|
FLOAT originY,
|
|
|
|
|
IDWriteInlineObject* inlineObject,
|
|
|
|
|
BOOL isSideways,
|
|
|
|
|
BOOL isRightToLeft,
|
|
|
|
|
IUnknown* clientDrawingEffect
|
|
|
|
|
)
|
|
|
|
|
{
|
|
|
|
|
return E_NOTIMPL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
STDMETHODIMP_(unsigned long) CustomTextRenderer::AddRef()
|
|
|
|
|
{
|
|
|
|
|
return InterlockedIncrement(&cRefCount_);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
STDMETHODIMP_(unsigned long) CustomTextRenderer::Release()
|
|
|
|
|
{
|
|
|
|
|
unsigned long newCount = InterlockedDecrement(&cRefCount_);
|
|
|
|
|
|
|
|
|
|
if (newCount == 0)
|
|
|
|
|
{
|
|
|
|
|
delete this;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return newCount;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
STDMETHODIMP CustomTextRenderer::IsPixelSnappingDisabled(
|
|
|
|
|
__maybenull void* clientDrawingContext,
|
|
|
|
|
__out BOOL* isDisabled
|
|
|
|
|
)
|
|
|
|
|
{
|
|
|
|
|
*isDisabled = FALSE;
|
|
|
|
|
return S_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
STDMETHODIMP CustomTextRenderer::GetCurrentTransform(
|
|
|
|
|
__maybenull void* clientDrawingContext,
|
|
|
|
|
__out DWRITE_MATRIX* transform
|
|
|
|
|
)
|
|
|
|
|
{
|
|
|
|
|
pRT_->GetTransform(reinterpret_cast<D2D1_MATRIX_3X2_F*>(transform));
|
|
|
|
|
return S_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
STDMETHODIMP CustomTextRenderer::GetPixelsPerDip(
|
|
|
|
|
__maybenull void* clientDrawingContext,
|
|
|
|
|
__out FLOAT* pixelsPerDip
|
|
|
|
|
)
|
|
|
|
|
{
|
|
|
|
|
float x, yUnused;
|
|
|
|
|
|
|
|
|
|
pRT_->GetDpi(&x, &yUnused);
|
|
|
|
|
*pixelsPerDip = x / 96;
|
|
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
STDMETHODIMP CustomTextRenderer::QueryInterface(
|
|
|
|
|
IID const& riid,
|
|
|
|
|
void** ppvObject
|
|
|
|
|
)
|
|
|
|
|
{
|
|
|
|
|
if (__uuidof(IDWriteTextRenderer) == riid)
|
|
|
|
|
{
|
|
|
|
|
*ppvObject = this;
|
|
|
|
|
}
|
|
|
|
|
else if (__uuidof(IDWritePixelSnapping) == riid)
|
|
|
|
|
{
|
|
|
|
|
*ppvObject = this;
|
|
|
|
|
}
|
|
|
|
|
else if (__uuidof(IUnknown) == riid)
|
|
|
|
|
{
|
|
|
|
|
*ppvObject = this;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2018-05-17 15:22:14 +08:00
|
|
|
*ppvObject = nullptr;
|
2018-04-21 23:09:59 +08:00
|
|
|
return E_FAIL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
AddRef();
|
|
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
|
}
|