Magic_Game/core/Node/Text.cpp

446 lines
8.0 KiB
C++
Raw Normal View History

2018-04-21 21:24:46 +08:00
#include "..\e2dnode.h"
//-------------------------------------------------------
// Style
//-------------------------------------------------------
e2d::Text::Style::Style()
2018-05-24 20:37:34 +08:00
: color(Color::White)
, alignment(Align::Left)
, wrapping(false)
, wrappingWidth(0.0)
, lineSpacing(0.0)
, hasUnderline(false)
, hasStrikethrough(false)
, hasOutline(true)
2018-05-24 20:37:34 +08:00
, outlineColor(Color(Color::Black, 0.5))
, outlineWidth(1.0)
2018-05-24 20:37:34 +08:00
, outlineJoin(LineJoin::Round)
{}
e2d::Text::Style::Style(
Color color,
Align alignment,
bool wrapping,
double wrappingWidth,
double lineSpacing,
bool hasUnderline,
bool hasStrikethrough,
2018-04-22 14:08:29 +08:00
bool hasOutline,
Color outlineColor,
double outlineWidth,
LineJoin outlineJoin
)
: color(color)
, alignment(alignment)
, wrapping(wrapping)
, wrappingWidth(wrappingWidth)
, lineSpacing(lineSpacing)
, hasUnderline(hasUnderline)
, hasStrikethrough(hasStrikethrough)
, hasOutline(hasOutline)
, outlineColor(outlineColor)
, outlineWidth(outlineWidth)
, outlineJoin(outlineJoin)
{}
//-------------------------------------------------------
// Text
//-------------------------------------------------------
e2d::Text::Text()
: _font()
, _style()
, _textLayout(nullptr)
, _textFormat(nullptr)
{
}
e2d::Text::Text(const String & text, const Font & font, const Style & style)
: _font(font)
, _style(style)
, _textLayout(nullptr)
, _textFormat(nullptr)
, _text(text)
{
_reset();
}
e2d::Text::~Text()
{
2018-05-22 12:24:43 +08:00
SafeRelease(_textFormat);
SafeRelease(_textLayout);
}
e2d::String e2d::Text::getText() const
{
return _text;
}
2018-05-22 22:00:47 +08:00
e2d::Font e2d::Text::getFont() const
{
return _font;
}
e2d::Text::Style e2d::Text::getStyle() const
{
return _style;
}
e2d::String e2d::Text::getFontFamily() const
{
return _font.family;
}
double e2d::Text::getFontSize() const
{
return _font.size;
}
UINT e2d::Text::getFontWeight() const
{
return _font.weight;
}
e2d::Color e2d::Text::getColor() const
{
return _style.color;
}
e2d::Color e2d::Text::getOutlineColor() const
{
return _style.outlineColor;
}
double e2d::Text::getOutlineWidth() const
{
return _style.outlineWidth;
}
e2d::LineJoin e2d::Text::getOutlineJoin() const
{
return _style.outlineJoin;
}
int e2d::Text::getLineCount() const
{
if (_textLayout)
2018-04-09 18:41:56 +08:00
{
DWRITE_TEXT_METRICS metrics;
_textLayout->GetMetrics(&metrics);
2018-04-09 18:41:56 +08:00
return static_cast<int>(metrics.lineCount);
}
else
{
return 0;
}
}
bool e2d::Text::isItalic() const
{
return _font.italic;
}
2018-04-22 14:08:29 +08:00
bool e2d::Text::hasStrikethrough() const
{
return _style.hasStrikethrough;
2018-04-22 14:08:29 +08:00
}
bool e2d::Text::hasUnderline() const
{
return _style.hasUnderline;
2018-04-22 14:08:29 +08:00
}
bool e2d::Text::hasOutline() const
{
return _style.hasOutline;
}
void e2d::Text::setText(const String& text)
{
_text = text;
_reset();
}
void e2d::Text::setStyle(const Style& style)
{
_style = style;
_reset();
}
void e2d::Text::setFont(const Font & font)
{
_font = font;
_reset();
}
void e2d::Text::setFontFamily(const String& family)
{
_font.family = family;
_reset();
}
void e2d::Text::setFontSize(double size)
{
_font.size = size;
_reset();
}
void e2d::Text::setFontWeight(UINT weight)
{
_font.weight = weight;
_reset();
}
void e2d::Text::setColor(Color color)
{
_style.color = color;
}
void e2d::Text::setItalic(bool value)
{
_font.italic = value;
_reset();
}
void e2d::Text::setWrapping(bool wrapping)
{
if (_style.wrapping != wrapping)
{
_style.wrapping = wrapping;
_reset();
}
}
2018-05-10 14:16:36 +08:00
void e2d::Text::setWrappingWidth(double wrappingWidth)
{
2018-05-10 14:16:36 +08:00
if (_style.wrappingWidth != wrappingWidth)
{
2018-07-04 12:49:05 +08:00
_style.wrappingWidth = std::max(wrappingWidth, 0.0);
if (_style.wrapping)
{
_reset();
}
}
}
2018-05-10 14:16:36 +08:00
void e2d::Text::setLineSpacing(double lineSpacing)
{
2018-05-10 14:16:36 +08:00
if (_style.lineSpacing != lineSpacing)
{
2018-05-10 14:16:36 +08:00
_style.lineSpacing = lineSpacing;
_reset();
}
}
void e2d::Text::setAlignment(Align align)
{
if (_style.alignment != align)
{
_style.alignment = align;
_reset();
}
}
void e2d::Text::setUnderline(bool hasUnderline)
{
if (_style.hasUnderline != hasUnderline)
{
_style.hasUnderline = hasUnderline;
if (!_textFormat)
_createFormat();
_createLayout();
}
}
void e2d::Text::setStrikethrough(bool hasStrikethrough)
{
if (_style.hasStrikethrough != hasStrikethrough)
{
_style.hasStrikethrough = hasStrikethrough;
if (!_textFormat)
_createFormat();
_createLayout();
}
}
2018-04-22 14:08:29 +08:00
void e2d::Text::setOutline(bool hasOutline)
{
_style.hasOutline = hasOutline;
}
void e2d::Text::setOutlineColor(Color outlineColor)
{
_style.outlineColor = outlineColor;
}
void e2d::Text::setOutlineWidth(double outlineWidth)
{
_style.outlineWidth = outlineWidth;
}
void e2d::Text::setOutlineJoin(LineJoin outlineJoin)
{
_style.outlineJoin = outlineJoin;
}
void e2d::Text::onRender()
{
if (_textLayout)
{
auto renderer = Renderer::getInstance();
// <20><><EFBFBD><EFBFBD><EFBFBD>ı<EFBFBD><C4B1><EFBFBD><EFBFBD><EFBFBD>
D2D1_RECT_F textLayoutRect = D2D1::RectF(0, 0, _width, _height);
// <20><><EFBFBD>û<EFBFBD>ˢ<EFBFBD><CBA2>ɫ<EFBFBD><C9AB>͸<EFBFBD><CDB8><EFBFBD><EFBFBD>
renderer->getSolidColorBrush()->SetOpacity(_displayOpacity);
// <20><>ȡ<EFBFBD>ı<EFBFBD><C4B1><EFBFBD>Ⱦ<EFBFBD><C8BE>
auto pTextRenderer = renderer->getTextRenderer();
pTextRenderer->SetTextStyle(
_style.color.toD2DColorF(),
_style.hasOutline,
_style.outlineColor.toD2DColorF(),
float(_style.outlineWidth),
D2D1_LINE_JOIN(_style.outlineJoin)
);
_textLayout->Draw(nullptr, pTextRenderer, 0, 0);
}
}
void e2d::Text::_reset()
{
// <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ָ<EFBFBD>ʽ<EFBFBD><CABD>
_createFormat();
// <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ֲ<EFBFBD><D6B2><EFBFBD>
_createLayout();
}
void e2d::Text::_createFormat()
{
2018-05-22 12:24:43 +08:00
SafeRelease(_textFormat);
HRESULT hr = Renderer::getWriteFactory()->CreateTextFormat(
(const WCHAR *)_font.family,
nullptr,
DWRITE_FONT_WEIGHT(_font.weight),
_font.italic ? DWRITE_FONT_STYLE_ITALIC : DWRITE_FONT_STYLE_NORMAL,
DWRITE_FONT_STRETCH_NORMAL,
float(_font.size),
2018-04-17 11:41:33 +08:00
L"",
&_textFormat
);
2018-05-24 12:24:39 +08:00
if (FAILED(hr))
{
WARN("Text::_createFormat error : Create IDWriteTextFormat failed!");
_textFormat = nullptr;
return;
}
else
{
// <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ֶ<EFBFBD><D6B6>ʽ
_textFormat->SetTextAlignment(DWRITE_TEXT_ALIGNMENT(_style.alignment));
// <20><><EFBFBD><EFBFBD><EFBFBD>м<EFBFBD><D0BC><EFBFBD>
if (_style.lineSpacing == 0.0)
{
_textFormat->SetLineSpacing(DWRITE_LINE_SPACING_METHOD_DEFAULT, 0, 0);
}
else
{
_textFormat->SetLineSpacing(
DWRITE_LINE_SPACING_METHOD_UNIFORM,
float(_style.lineSpacing),
float(_style.lineSpacing) * 0.8f
);
}
// <20><><EFBFBD><EFBFBD><EFBFBD>ı<EFBFBD><C4B1>Զ<EFBFBD><D4B6><EFBFBD><EFBFBD><EFBFBD>ʱ<EFBFBD><CAB1><EFBFBD><EFBFBD><EFBFBD>û<EFBFBD><C3BB><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
if (_style.wrapping)
{
_textFormat->SetWordWrapping(DWRITE_WORD_WRAPPING_WRAP);
}
else
{
_textFormat->SetWordWrapping(DWRITE_WORD_WRAPPING_NO_WRAP);
}
}
}
void e2d::Text::_createLayout()
{
2018-05-22 12:24:43 +08:00
SafeRelease(_textLayout);
// <20>ı<EFBFBD>Ϊ<EFBFBD><CEAA><EFBFBD>ַ<EFBFBD><D6B7><EFBFBD>ʱ<EFBFBD><CAB1><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
if (_text.isEmpty())
{
this->setSize(0, 0);
return;
}
if (_textFormat == nullptr)
{
2018-05-24 12:24:39 +08:00
WARN("Text::_createLayout failed! _textFormat NULL pointer exception.");
return;
}
UINT32 length = (UINT32)_text.getLength();
// <20><><EFBFBD><EFBFBD> TextLayout
HRESULT hr;
// <20><><EFBFBD>ı<EFBFBD><C4B1>Զ<EFBFBD><D4B6><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>½<EFBFBD><C2BD>д<EFBFBD><D0B4><EFBFBD>
if (_style.wrapping)
{
hr = Renderer::getWriteFactory()->CreateTextLayout(
(const WCHAR *)_text,
length,
_textFormat,
float(_style.wrappingWidth),
0,
&_textLayout
);
if (_textLayout)
{
// <20><>ȡ<EFBFBD>ı<EFBFBD><C4B1><EFBFBD><EFBFBD>ֵĿ<D6B5><C4BF>Ⱥ͸߶<CDB8>
DWRITE_TEXT_METRICS metrics;
_textLayout->GetMetrics(&metrics);
// <20><><EFBFBD><EFBFBD><EFBFBD>ı<EFBFBD><C4B1><EFBFBD><EFBFBD><EFBFBD>
this->setSize(metrics.layoutWidth, metrics.height);
}
}
else
{
hr = Renderer::getWriteFactory()->CreateTextLayout((const WCHAR *)_text, length, _textFormat, 0, 0, &_textLayout);
// Ϊ<><CEAA>ֹ<EFBFBD>ı<EFBFBD><C4B1><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><E2A3AC><EFBFBD>ݸղŴ<D5B2><C5B4><EFBFBD><EFBFBD><EFBFBD> layout <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>´<EFBFBD><C2B4><EFBFBD><EFBFBD><EFBFBD>
if (_textLayout)
{
// <20><>ȡ<EFBFBD>ı<EFBFBD><C4B1><EFBFBD><EFBFBD>ֵĿ<D6B5><C4BF>Ⱥ͸߶<CDB8>
DWRITE_TEXT_METRICS metrics;
_textLayout->GetMetrics(&metrics);
// <20><><EFBFBD><EFBFBD><EFBFBD>ı<EFBFBD><C4B1><EFBFBD><EFBFBD><EFBFBD>
this->setSize(metrics.width, metrics.height);
// <20><><EFBFBD>´<EFBFBD><C2B4><EFBFBD> layout
2018-05-22 12:24:43 +08:00
SafeRelease(_textLayout);
hr = Renderer::getWriteFactory()->CreateTextLayout((const WCHAR *)_text, length, _textFormat, _width, 0, &_textLayout);
}
}
2018-05-24 12:24:39 +08:00
if (FAILED(hr))
{
WARN("Text::_createLayout error : Create IDWriteTextLayout failed!");
_textLayout = nullptr;
return;
}
// <20><><EFBFBD><EFBFBD><EFBFBD>»<EFBFBD><C2BB>ߺ<EFBFBD>ɾ<EFBFBD><C9BE><EFBFBD><EFBFBD>
2018-04-01 13:28:32 +08:00
DWRITE_TEXT_RANGE range = { 0, length };
if (_style.hasUnderline)
{
_textLayout->SetUnderline(true, range);
}
if (_style.hasStrikethrough)
{
_textLayout->SetStrikethrough(true, range);
}
}