Add a 'subpixel glyph positions' option to UI (#1953)
When this option is enabled, text will be formatted with subpixel (fractional) positions on the x-axis. Positions on the y-axis are still pixel-aligned. Note that this option has no effect if the hinting level is set to FONT_HINT_LEVEL_NORMAL, as each glyph is rounded to an integral pixel size by the hinter. It only makes a different if the hinting mode is LIGHT (vertical hinting only) or NONE. With subpixel positioning, the output will look blurrier due to texture filtering. TODO: Add horizontal oversampling to improve sharpness. That needs extra memory so it should be configurable.
This commit is contained in:
parent
7fa6f9c960
commit
e603eed814
@ -81,10 +81,6 @@ void Typography::Start()
|
||||
CreateCheckbox("White background", URHO3D_HANDLER(Typography, HandleWhiteBackground))
|
||||
->SetChecked(false);
|
||||
|
||||
// Add a checkbox for the global ForceAutoHint setting. This affects character spacing.
|
||||
CreateCheckbox("UI::SetForceAutoHint", URHO3D_HANDLER(Typography, HandleForceAutoHint))
|
||||
->SetChecked(ui->GetForceAutoHint());
|
||||
|
||||
// Add a checkbox to toggle SRGB output conversion (if available).
|
||||
// This will give more correct text output for FreeType fonts, as the FreeType rasterizer
|
||||
// outputs linear coverage values rather than SRGB values. However, this feature isn't
|
||||
@ -92,6 +88,14 @@ void Typography::Start()
|
||||
CreateCheckbox("Graphics::SetSRGB", URHO3D_HANDLER(Typography, HandleSRGB))
|
||||
->SetChecked(GetSubsystem<Graphics>()->GetSRGB());
|
||||
|
||||
// Add a checkbox for the global ForceAutoHint setting. This affects character spacing.
|
||||
CreateCheckbox("UI::SetForceAutoHint", URHO3D_HANDLER(Typography, HandleForceAutoHint))
|
||||
->SetChecked(ui->GetForceAutoHint());
|
||||
|
||||
// Add a checkbox for the global SubpixelGlyphPositions setting. This affects character spacing.
|
||||
CreateCheckbox("UI::SetSubpixelGlyphPositions", URHO3D_HANDLER(Typography, HandleSubpixelGlyphPositions))
|
||||
->SetChecked(ui->GetSubpixelGlyphPositions());
|
||||
|
||||
// Add a drop-down menu to control the font hinting level.
|
||||
const char* items[] = {
|
||||
"FONT_HINT_LEVEL_NONE",
|
||||
@ -225,10 +229,16 @@ void Typography::HandleSRGB(StringHash eventType, VariantMap& eventData)
|
||||
}
|
||||
}
|
||||
|
||||
void Typography::HandleSubpixelGlyphPositions(StringHash eventType, VariantMap& eventData)
|
||||
{
|
||||
CheckBox* box = static_cast<CheckBox*>(eventData[Toggled::P_ELEMENT].GetPtr());
|
||||
bool checked = box->IsChecked();
|
||||
GetSubsystem<UI>()->SetSubpixelGlyphPositions(checked);
|
||||
}
|
||||
|
||||
void Typography::HandleFontHintLevel(StringHash eventType, VariantMap& eventData)
|
||||
{
|
||||
DropDownList* list = static_cast<DropDownList*>(eventData[Toggled::P_ELEMENT].GetPtr());
|
||||
unsigned i = list->GetSelection();
|
||||
|
||||
GetSubsystem<UI>()->SetFontHintLevel((FontHintLevel)i);
|
||||
}
|
||||
|
@ -58,4 +58,5 @@ private:
|
||||
void HandleSRGB(StringHash eventType, VariantMap& eventData);
|
||||
void HandleForceAutoHint(StringHash eventType, VariantMap& eventData);
|
||||
void HandleFontHintLevel(StringHash eventType, VariantMap& eventData);
|
||||
void HandleSubpixelGlyphPositions(StringHash eventType, VariantMap& eventData);
|
||||
};
|
||||
|
@ -76,7 +76,7 @@ static void RegisterFont(asIScriptEngine* engine)
|
||||
engine->RegisterObjectMethod("Font", "bool SaveXML(File@+, int, bool usedGlyphs = false, const String&in indentation = \"\t\")", asFUNCTION(FontSaveXMLFile), asCALL_CDECL_OBJLAST);
|
||||
engine->RegisterObjectMethod("Font", "bool SaveXML(VectorBuffer&, int, bool usedGlyphs = false, const String&in indentation = \"\t\")", asFUNCTION(FontSaveXMLVectorBuffer), asCALL_CDECL_OBJLAST);
|
||||
engine->RegisterObjectMethod("Font", "bool SaveXML(const String&in, int, bool usedGlyphs = false, const String&in indentation = \"\t\")", asFUNCTION(FontSaveXML), asCALL_CDECL_OBJLAST);
|
||||
engine->RegisterObjectMethod("Font", "IntVector2 GetTotalGlyphOffset(int) const", asMETHOD(Font, GetTotalGlyphOffset), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Font", "IntVector2 GetTotalGlyphOffset(float) const", asMETHOD(Font, GetTotalGlyphOffset), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Font", "void set_absoluteGlyphOffset(const IntVector2&)", asMETHOD(Font, SetAbsoluteGlyphOffset), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Font", "const IntVector2& get_absoluteGlyphOffset() const", asMETHOD(Font, GetAbsoluteGlyphOffset), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Font", "void set_scaledGlyphOffset(const Vector2&)", asMETHOD(Font, SetScaledGlyphOffset), asCALL_THISCALL);
|
||||
@ -411,10 +411,10 @@ static void RegisterText(asIScriptEngine* engine)
|
||||
engine->RegisterObjectMethod("Text", "const Color& get_effectColor() const", asMETHOD(Text, GetEffectColor), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text", "uint get_numRows() const", asMETHOD(Text, GetNumRows), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text", "uint get_numChars() const", asMETHOD(Text, GetNumChars), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text", "int get_rowWidths(uint) const", asMETHOD(Text, GetRowWidth), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text", "IntVector2 get_charPositions(uint)", asMETHOD(Text, GetCharPosition), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text", "IntVector2 get_charSizes(uint)", asMETHOD(Text, GetCharSize), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text", "int get_rowHeight() const", asMETHOD(Text, GetRowHeight), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text", "float get_rowWidths(uint) const", asMETHOD(Text, GetRowWidth), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text", "Vector2 get_charPositions(uint)", asMETHOD(Text, GetCharPosition), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text", "Vector2 get_charSizes(uint)", asMETHOD(Text, GetCharSize), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text", "float get_rowHeight() const", asMETHOD(Text, GetRowHeight), asCALL_THISCALL);
|
||||
}
|
||||
|
||||
static void RegisterText3D(asIScriptEngine* engine)
|
||||
@ -466,10 +466,10 @@ static void RegisterText3D(asIScriptEngine* engine)
|
||||
engine->RegisterObjectMethod("Text3D", "FaceCameraMode get_faceCameraMode() const", asMETHOD(Text3D, GetFaceCameraMode), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text3D", "uint get_numRows() const", asMETHOD(Text3D, GetNumRows), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text3D", "uint get_numChars() const", asMETHOD(Text3D, GetNumChars), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text3D", "int get_rowWidths(uint) const", asMETHOD(Text3D, GetRowWidth), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text3D", "IntVector2 get_charPositions(uint)", asMETHOD(Text3D, GetCharPosition), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text3D", "IntVector2 get_charSizes(uint)", asMETHOD(Text3D, GetCharSize), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text3D", "int get_rowHeight() const", asMETHOD(Text3D, GetRowHeight), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text3D", "float get_rowWidths(uint) const", asMETHOD(Text3D, GetRowWidth), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text3D", "Vector2 get_charPositions(uint)", asMETHOD(Text3D, GetCharPosition), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text3D", "Vector2 get_charSizes(uint)", asMETHOD(Text3D, GetCharSize), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("Text3D", "float get_rowHeight() const", asMETHOD(Text3D, GetRowHeight), asCALL_THISCALL);
|
||||
}
|
||||
|
||||
static void RegisterLineEdit(asIScriptEngine* engine)
|
||||
@ -789,6 +789,8 @@ static void RegisterUI(asIScriptEngine* engine)
|
||||
engine->RegisterObjectMethod("UI", "bool get_forceAutoHint() const", asMETHOD(UI, GetForceAutoHint), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("UI", "void set_fontHintLevel(FontHintLevel)", asMETHOD(UI, SetFontHintLevel), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("UI", "FontHintLevel get_fontHintLevel() const", asMETHOD(UI, GetFontHintLevel), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("UI", "void set_subpixelGlyphPositions(bool)", asMETHOD(UI, SetSubpixelGlyphPositions), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("UI", "bool get_subpixelGlyphPositions() const", asMETHOD(UI, GetSubpixelGlyphPositions), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("UI", "void set_scale(float value)", asMETHOD(UI, SetScale), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("UI", "float get_scale() const", asMETHOD(UI, GetScale), asCALL_THISCALL);
|
||||
engine->RegisterObjectMethod("UI", "void set_customSize(const IntVector2&in)", asMETHODPR(UI, SetCustomSize, (const IntVector2&), void), asCALL_THISCALL);
|
||||
|
@ -15,7 +15,7 @@ class Font : public Resource
|
||||
|
||||
const IntVector2& GetAbsoluteGlyphOffset() const;
|
||||
const Vector2& GetScaledGlyphOffset() const;
|
||||
IntVector2 GetTotalGlyphOffset(int pointSize) const;
|
||||
IntVector2 GetTotalGlyphOffset(float pointSize) const;
|
||||
FontType GetFontType() const;
|
||||
bool IsSDFFont() const;
|
||||
|
||||
|
@ -49,12 +49,12 @@ class Text : public UIElement
|
||||
int GetEffectStrokeThickness() const;
|
||||
bool GetEffectRoundStroke() const;
|
||||
const Color& GetEffectColor() const;
|
||||
int GetRowHeight() const;
|
||||
float GetRowHeight() const;
|
||||
unsigned GetNumRows() const;
|
||||
unsigned GetNumChars() const;
|
||||
int GetRowWidth(unsigned index) const;
|
||||
IntVector2 GetCharPosition(unsigned index);
|
||||
IntVector2 GetCharSize(unsigned index);
|
||||
float GetRowWidth(unsigned index) const;
|
||||
Vector2 GetCharPosition(unsigned index);
|
||||
Vector2 GetCharSize(unsigned index);
|
||||
|
||||
void SetEffectDepthBias(float bias);
|
||||
float GetEffectDepthBias() const;
|
||||
@ -75,7 +75,7 @@ class Text : public UIElement
|
||||
tolua_property__get_set int effectStrokeThickness;
|
||||
tolua_property__get_set bool effectRoundStroke;
|
||||
tolua_property__get_set Color& effectColor;
|
||||
tolua_readonly tolua_property__get_set int rowHeight;
|
||||
tolua_readonly tolua_property__get_set float rowHeight;
|
||||
tolua_readonly tolua_property__get_set unsigned numRows;
|
||||
tolua_readonly tolua_property__get_set unsigned numChars;
|
||||
};
|
||||
|
@ -61,12 +61,12 @@ class Text3D : public Drawable
|
||||
float GetEffectDepthBias() const;
|
||||
int GetWidth() const;
|
||||
int GetHeight() const;
|
||||
int GetRowHeight() const;
|
||||
float GetRowHeight() const;
|
||||
unsigned GetNumRows() const;
|
||||
unsigned GetNumChars() const;
|
||||
int GetRowWidth(unsigned index) const;
|
||||
IntVector2 GetCharPosition(unsigned index);
|
||||
IntVector2 GetCharSize(unsigned index);
|
||||
float GetRowWidth(unsigned index) const;
|
||||
Vector2 GetCharPosition(unsigned index);
|
||||
Vector2 GetCharSize(unsigned index);
|
||||
const Color& GetColor(Corner corner) const;
|
||||
float GetOpacity() const;
|
||||
bool IsFixedScreenSize() const;
|
||||
@ -90,7 +90,7 @@ class Text3D : public Drawable
|
||||
tolua_property__get_set int width;
|
||||
tolua_property__get_set Color& color; // Write only property.
|
||||
tolua_readonly tolua_property__get_set int height;
|
||||
tolua_readonly tolua_property__get_set int rowHeight;
|
||||
tolua_readonly tolua_property__get_set float rowHeight;
|
||||
tolua_readonly tolua_property__get_set unsigned numRows;
|
||||
tolua_readonly tolua_property__get_set unsigned numChars;
|
||||
tolua_property__get_set float opacity;
|
||||
|
@ -31,6 +31,7 @@ class UI : public Object
|
||||
void SetUseMutableGlyphs(bool enable);
|
||||
void SetForceAutoHint(bool enable);
|
||||
void SetFontHintLevel(FontHintLevel level);
|
||||
void SetSubpixelGlyphPositions(bool enable);
|
||||
void SetScale(float scale);
|
||||
void SetWidth(float width);
|
||||
void SetHeight(float height);
|
||||
@ -59,6 +60,7 @@ class UI : public Object
|
||||
bool GetUseMutableGlyphs() const;
|
||||
bool GetForceAutoHint() const;
|
||||
FontHintLevel GetFontHintLevel() const;
|
||||
bool GetSubpixelGlyphPositions() const;
|
||||
bool HasModalElement() const;
|
||||
bool IsDragging() const;
|
||||
float GetScale() const;
|
||||
@ -82,6 +84,8 @@ class UI : public Object
|
||||
tolua_property__get_set bool useScreenKeyboard;
|
||||
tolua_property__get_set bool useMutableGlyphs;
|
||||
tolua_property__get_set bool forceAutoHint;
|
||||
tolua_property__get_set FontHintLevel fontHintLevel;
|
||||
tolua_property__get_set bool subpixelGlyphPositions;
|
||||
tolua_readonly tolua_property__has_set bool modalElement;
|
||||
tolua_property__get_set float scale;
|
||||
tolua_property__get_set IntVector2& customSize;
|
||||
|
@ -178,10 +178,10 @@ FontFace* Font::GetFace(float pointSize)
|
||||
}
|
||||
}
|
||||
|
||||
IntVector2 Font::GetTotalGlyphOffset(int pointSize) const
|
||||
IntVector2 Font::GetTotalGlyphOffset(float pointSize) const
|
||||
{
|
||||
Vector2 multipliedOffset = (float)pointSize * scaledOffset_;
|
||||
return absoluteOffset_ + IntVector2((int)multipliedOffset.x_, (int)multipliedOffset.y_);
|
||||
Vector2 multipliedOffset = pointSize * scaledOffset_;
|
||||
return absoluteOffset_ + IntVector2((int)(multipliedOffset.x_ + 0.5f), (int)(multipliedOffset.y_ + 0.5f));
|
||||
}
|
||||
|
||||
void Font::ReleaseFaces()
|
||||
|
@ -80,7 +80,7 @@ public:
|
||||
const Vector2& GetScaledGlyphOffset() const { return scaledOffset_; }
|
||||
|
||||
/// Return the total effective offset for a point size.
|
||||
IntVector2 GetTotalGlyphOffset(int pointSize) const;
|
||||
IntVector2 GetTotalGlyphOffset(float pointSize) const;
|
||||
|
||||
/// Release font faces and recreate them next time when requested. Called when font textures lost or global font properties change.
|
||||
void ReleaseFaces();
|
||||
|
@ -70,7 +70,7 @@ const FontGlyph* FontFace::GetGlyph(unsigned c)
|
||||
return 0;
|
||||
}
|
||||
|
||||
short FontFace::GetKerning(unsigned c, unsigned d) const
|
||||
float FontFace::GetKerning(unsigned c, unsigned d) const
|
||||
{
|
||||
if (kerningMapping_.Empty())
|
||||
return 0;
|
||||
@ -83,7 +83,7 @@ short FontFace::GetKerning(unsigned c, unsigned d) const
|
||||
|
||||
unsigned value = (c << 16) + d;
|
||||
|
||||
HashMap<unsigned, short>::ConstIterator i = kerningMapping_.Find(value);
|
||||
HashMap<unsigned, float>::ConstIterator i = kerningMapping_.Find(value);
|
||||
if (i != kerningMapping_.End())
|
||||
return i->second_;
|
||||
|
||||
|
@ -52,7 +52,7 @@ struct URHO3D_API FontGlyph
|
||||
/// Glyph Y offset from origin.
|
||||
short offsetY_;
|
||||
/// Horizontal advance.
|
||||
short advanceX_;
|
||||
float advanceX_;
|
||||
/// Texture page. M_MAX_UNSIGNED if not yet resident on any texture.
|
||||
unsigned page_;
|
||||
/// Used flag.
|
||||
@ -79,7 +79,7 @@ public:
|
||||
virtual bool HasMutableGlyphs() const { return false; }
|
||||
|
||||
/// Return the kerning for a character and the next character.
|
||||
short GetKerning(unsigned c, unsigned d) const;
|
||||
float GetKerning(unsigned c, unsigned d) const;
|
||||
/// Return true when one of the texture has a data loss.
|
||||
bool IsDataLost() const;
|
||||
|
||||
@ -87,7 +87,7 @@ public:
|
||||
float GetPointSize() const { return pointSize_; }
|
||||
|
||||
/// Return row height.
|
||||
int GetRowHeight() const { return rowHeight_; }
|
||||
float GetRowHeight() const { return rowHeight_; }
|
||||
|
||||
/// Return textures.
|
||||
const Vector<SharedPtr<Texture2D> >& GetTextures() const { return textures_; }
|
||||
@ -104,13 +104,13 @@ protected:
|
||||
/// Glyph mapping.
|
||||
HashMap<unsigned, FontGlyph> glyphMapping_;
|
||||
/// Kerning mapping.
|
||||
HashMap<unsigned, short> kerningMapping_;
|
||||
HashMap<unsigned, float> kerningMapping_;
|
||||
/// Glyph texture pages.
|
||||
Vector<SharedPtr<Texture2D> > textures_;
|
||||
/// Point size.
|
||||
float pointSize_;
|
||||
/// Row height.
|
||||
int rowHeight_;
|
||||
float rowHeight_;
|
||||
};
|
||||
|
||||
}
|
||||
|
@ -251,7 +251,7 @@ bool FontFaceBitmap::Load(FontFace* fontFace, bool usedGlyphs)
|
||||
for (unsigned i = 0; i < newImages.Size(); ++i)
|
||||
textures_[i] = LoadFaceTexture(newImages[i]);
|
||||
|
||||
for (HashMap<unsigned, short>::ConstIterator i = fontFace->kerningMapping_.Begin(); i != fontFace->kerningMapping_.End(); ++i)
|
||||
for (HashMap<unsigned, float>::ConstIterator i = fontFace->kerningMapping_.Begin(); i != fontFace->kerningMapping_.End(); ++i)
|
||||
{
|
||||
unsigned first = (i->first_) >> 16;
|
||||
unsigned second = (i->first_) & 0xffff;
|
||||
@ -329,7 +329,7 @@ bool FontFaceBitmap::Save(Serializer& dest, int pointSize, const String& indenta
|
||||
if (!kerningMapping_.Empty())
|
||||
{
|
||||
XMLElement kerningsElem = rootElem.CreateChild("kernings");
|
||||
for (HashMap<unsigned, short>::ConstIterator i = kerningMapping_.Begin(); i != kerningMapping_.End(); ++i)
|
||||
for (HashMap<unsigned, float>::ConstIterator i = kerningMapping_.Begin(); i != kerningMapping_.End(); ++i)
|
||||
{
|
||||
XMLElement kerningElem = kerningsElem.CreateChild("kerning");
|
||||
kerningElem.SetInt("first", i->first_ >> 16);
|
||||
|
@ -41,9 +41,9 @@
|
||||
namespace Urho3D
|
||||
{
|
||||
|
||||
inline int RoundToPixels(FT_Pos value)
|
||||
inline float FixedToFloat(FT_Pos value)
|
||||
{
|
||||
return (int)(value >> 6) + (((value & 0x3f) >= 0x20) ? 1 : 0);
|
||||
return value / 64.0f;
|
||||
}
|
||||
|
||||
/// FreeType library subsystem.
|
||||
@ -172,19 +172,20 @@ bool FontFaceFreeType::Load(const unsigned char* fontData, unsigned fontDataSize
|
||||
loadMode_ |= FT_LOAD_TARGET_LIGHT;
|
||||
}
|
||||
|
||||
ascender_ = RoundToPixels(face->size->metrics.ascender);
|
||||
rowHeight_ = RoundToPixels(face->size->metrics.height);
|
||||
ascender_ = FixedToFloat(face->size->metrics.ascender);
|
||||
rowHeight_ = FixedToFloat(face->size->metrics.height);
|
||||
pointSize_ = pointSize;
|
||||
|
||||
// Check if the font's OS/2 info gives different (larger) values for ascender & descender
|
||||
TT_OS2* os2Info = (TT_OS2*)FT_Get_Sfnt_Table(face, ft_sfnt_os2);
|
||||
if (os2Info)
|
||||
{
|
||||
int descender = RoundToPixels(face->size->metrics.descender);
|
||||
ascender_ = Max(ascender_, os2Info->usWinAscent * face->size->metrics.y_ppem / face->units_per_EM);
|
||||
ascender_ = Max(ascender_, os2Info->sTypoAscender * face->size->metrics.y_ppem / face->units_per_EM);
|
||||
descender = Max(descender, os2Info->usWinDescent * face->size->metrics.y_ppem / face->units_per_EM);
|
||||
descender = Max(descender, os2Info->sTypoDescender * face->size->metrics.y_ppem / face->units_per_EM);
|
||||
float descender = FixedToFloat(face->size->metrics.descender);
|
||||
float unitsPerEm = face->units_per_EM;
|
||||
ascender_ = Max(ascender_, os2Info->usWinAscent * face->size->metrics.y_ppem / unitsPerEm);
|
||||
ascender_ = Max(ascender_, os2Info->sTypoAscender * face->size->metrics.y_ppem / unitsPerEm);
|
||||
descender = Max(descender, os2Info->usWinDescent * face->size->metrics.y_ppem / unitsPerEm);
|
||||
descender = Max(descender, os2Info->sTypoDescender * face->size->metrics.y_ppem / unitsPerEm);
|
||||
rowHeight_ = Max(rowHeight_, ascender_ + descender);
|
||||
}
|
||||
|
||||
@ -265,7 +266,7 @@ bool FontFaceFreeType::Load(const unsigned char* fontData, unsigned fontDataSize
|
||||
{
|
||||
unsigned leftIndex = deserializer.ReadUShort();
|
||||
unsigned rightIndex = deserializer.ReadUShort();
|
||||
short amount = RoundToPixels(deserializer.ReadShort());
|
||||
short amount = FixedToFloat(deserializer.ReadShort());
|
||||
|
||||
unsigned leftCharCode = leftIndex < numGlyphs ? charCodes[leftIndex] : 0;
|
||||
unsigned rightCharCode = rightIndex < numGlyphs ? charCodes[rightIndex] : 0;
|
||||
@ -367,7 +368,20 @@ bool FontFaceFreeType::LoadCharGlyph(unsigned charCode, Image* image)
|
||||
fontGlyph.height_ = slot->bitmap.rows;
|
||||
fontGlyph.offsetX_ = slot->bitmap_left;
|
||||
fontGlyph.offsetY_ = ascender_ - slot->bitmap_top;
|
||||
fontGlyph.advanceX_ = (short)RoundToPixels(slot->metrics.horiAdvance);
|
||||
|
||||
UI* ui = font_->GetSubsystem<UI>();
|
||||
FontHintLevel level = ui->GetFontHintLevel();
|
||||
bool subpixel = ui->GetSubpixelGlyphPositions();
|
||||
if (level <= FONT_HINT_LEVEL_LIGHT && subpixel && slot->linearHoriAdvance)
|
||||
{
|
||||
// linearHoriAdvance is stored in 16.16 fixed point, not the usual 26.6
|
||||
fontGlyph.advanceX_ = slot->linearHoriAdvance / 65536.0;
|
||||
}
|
||||
else
|
||||
{
|
||||
// Round to nearest pixel (only necessary when hinting is disabled)
|
||||
fontGlyph.advanceX_ = floor(FixedToFloat(slot->metrics.horiAdvance) + 0.5f);
|
||||
}
|
||||
}
|
||||
|
||||
int x = 0, y = 0;
|
||||
|
@ -60,7 +60,7 @@ private:
|
||||
/// Load mode.
|
||||
int loadMode_;
|
||||
/// Ascender.
|
||||
int ascender_;
|
||||
float ascender_;
|
||||
/// Has mutable glyph.
|
||||
bool hasMutableGlyph_;
|
||||
/// Glyph area allocator.
|
||||
|
@ -159,12 +159,12 @@ void Text::GetBatches(PODVector<UIBatch>& batches, PODVector<float>& vertexData,
|
||||
UIBatch batch(this, BLEND_ALPHA, currentScissor, 0, &vertexData);
|
||||
batch.SetColor(selectionColor_);
|
||||
|
||||
IntVector2 currentStart = charLocations_[selectionStart_].position_;
|
||||
IntVector2 currentEnd = currentStart;
|
||||
Vector2 currentStart = charLocations_[selectionStart_].position_;
|
||||
Vector2 currentEnd = currentStart;
|
||||
for (unsigned i = selectionStart_; i < selectionStart_ + selectionLength_; ++i)
|
||||
{
|
||||
// Check if row changes, and start a new quad in that case
|
||||
if (charLocations_[i].size_ != IntVector2::ZERO)
|
||||
if (charLocations_[i].size_ != Vector2::ZERO)
|
||||
{
|
||||
if (charLocations_[i].position_.y_ != currentStart.y_)
|
||||
{
|
||||
@ -223,7 +223,7 @@ void Text::GetBatches(PODVector<UIBatch>& batches, PODVector<float>& vertexData,
|
||||
{
|
||||
float x = Cos(angle * i) * floatThickness;
|
||||
float y = Sin(angle * i) * floatThickness;
|
||||
ConstructBatch(pageBatch, pageGlyphLocation, (int)x, (int)y, &effectColor_, effectDepthBias_);
|
||||
ConstructBatch(pageBatch, pageGlyphLocation, x, y, &effectColor_, effectDepthBias_);
|
||||
}
|
||||
}
|
||||
else
|
||||
@ -438,29 +438,29 @@ void Text::SetEffectDepthBias(float bias)
|
||||
effectDepthBias_ = bias;
|
||||
}
|
||||
|
||||
int Text::GetRowWidth(unsigned index) const
|
||||
float Text::GetRowWidth(unsigned index) const
|
||||
{
|
||||
return index < rowWidths_.Size() ? rowWidths_[index] : 0;
|
||||
}
|
||||
|
||||
IntVector2 Text::GetCharPosition(unsigned index)
|
||||
Vector2 Text::GetCharPosition(unsigned index)
|
||||
{
|
||||
if (charLocationsDirty_)
|
||||
UpdateCharLocations();
|
||||
if (charLocations_.Empty())
|
||||
return IntVector2::ZERO;
|
||||
return Vector2::ZERO;
|
||||
// For convenience, return the position of the text ending if index exceeded
|
||||
if (index > charLocations_.Size() - 1)
|
||||
index = charLocations_.Size() - 1;
|
||||
return charLocations_[index].position_;
|
||||
}
|
||||
|
||||
IntVector2 Text::GetCharSize(unsigned index)
|
||||
Vector2 Text::GetCharSize(unsigned index)
|
||||
{
|
||||
if (charLocationsDirty_)
|
||||
UpdateCharLocations();
|
||||
if (charLocations_.Size() < 2)
|
||||
return IntVector2::ZERO;
|
||||
return Vector2::ZERO;
|
||||
// For convenience, return the size of the last char if index exceeded (last size entry is zero)
|
||||
if (index > charLocations_.Size() - 2)
|
||||
index = charLocations_.Size() - 2;
|
||||
@ -527,7 +527,7 @@ void Text::UpdateText(bool onResize)
|
||||
int width = 0;
|
||||
int height = 0;
|
||||
int rowWidth = 0;
|
||||
int rowHeight = (int)(rowSpacing_ * rowHeight_);
|
||||
int rowHeight = (int)(rowSpacing_ * rowHeight_ + 0.5f);
|
||||
|
||||
// First see if the text must be split up
|
||||
if (!wordWrap_)
|
||||
@ -706,7 +706,7 @@ void Text::UpdateCharLocations()
|
||||
return;
|
||||
fontFace_ = face;
|
||||
|
||||
int rowHeight = (int)(rowSpacing_ * rowHeight_);
|
||||
int rowHeight = (int)(rowSpacing_ * rowHeight_ + 0.5f);
|
||||
|
||||
// Store position & size of each character, and locations per texture page
|
||||
unsigned numChars = unicodeText_.Size();
|
||||
@ -719,19 +719,19 @@ void Text::UpdateCharLocations()
|
||||
|
||||
unsigned rowIndex = 0;
|
||||
unsigned lastFilled = 0;
|
||||
int x = GetRowStartPosition(rowIndex) + offset.x_;
|
||||
int y = offset.y_;
|
||||
float x = floor(GetRowStartPosition(rowIndex) + offset.x_ + 0.5f);
|
||||
float y = floor(offset.y_ + 0.5f);
|
||||
|
||||
for (unsigned i = 0; i < printText_.Size(); ++i)
|
||||
{
|
||||
CharLocation loc;
|
||||
loc.position_ = IntVector2(x, y);
|
||||
loc.position_ = Vector2(x, y);
|
||||
|
||||
unsigned c = printText_[i];
|
||||
if (c != '\n')
|
||||
{
|
||||
const FontGlyph* glyph = face->GetGlyph(c);
|
||||
loc.size_ = IntVector2(glyph ? glyph->advanceX_ : 0, rowHeight_);
|
||||
loc.size_ = Vector2(glyph ? glyph->advanceX_ : 0, rowHeight_);
|
||||
if (glyph)
|
||||
{
|
||||
// Store glyph's location for rendering. Verify that glyph page is valid
|
||||
@ -744,7 +744,7 @@ void Text::UpdateCharLocations()
|
||||
}
|
||||
else
|
||||
{
|
||||
loc.size_ = IntVector2::ZERO;
|
||||
loc.size_ = Vector2::ZERO;
|
||||
x = GetRowStartPosition(++rowIndex);
|
||||
y += rowHeight;
|
||||
}
|
||||
@ -758,8 +758,8 @@ void Text::UpdateCharLocations()
|
||||
lastFilled = printToText_[i] + 1;
|
||||
}
|
||||
// Store the ending position
|
||||
charLocations_[numChars].position_ = IntVector2(x, y);
|
||||
charLocations_[numChars].size_ = IntVector2::ZERO;
|
||||
charLocations_[numChars].position_ = Vector2(x, y);
|
||||
charLocations_[numChars].size_ = Vector2::ZERO;
|
||||
|
||||
charLocationsDirty_ = false;
|
||||
}
|
||||
@ -784,7 +784,7 @@ void Text::ValidateSelection()
|
||||
|
||||
int Text::GetRowStartPosition(unsigned rowIndex) const
|
||||
{
|
||||
int rowWidth = 0;
|
||||
float rowWidth = 0;
|
||||
|
||||
if (rowIndex < rowWidths_.Size())
|
||||
rowWidth = rowWidths_[rowIndex];
|
||||
@ -806,7 +806,7 @@ int Text::GetRowStartPosition(unsigned rowIndex) const
|
||||
return ret;
|
||||
}
|
||||
|
||||
void Text::ConstructBatch(UIBatch& pageBatch, const PODVector<GlyphLocation>& pageGlyphLocation, int dx, int dy, Color* color,
|
||||
void Text::ConstructBatch(UIBatch& pageBatch, const PODVector<GlyphLocation>& pageGlyphLocation, float dx, float dy, Color* color,
|
||||
float depthBias)
|
||||
{
|
||||
unsigned startDataSize = pageBatch.vertexData_->Size();
|
||||
|
@ -45,16 +45,16 @@ enum TextEffect
|
||||
struct CharLocation
|
||||
{
|
||||
/// Position.
|
||||
IntVector2 position_;
|
||||
Vector2 position_;
|
||||
/// Size.
|
||||
IntVector2 size_;
|
||||
Vector2 size_;
|
||||
};
|
||||
|
||||
/// Glyph and its location within the text. Used when preparing text rendering.
|
||||
struct GlyphLocation
|
||||
{
|
||||
/// Construct.
|
||||
GlyphLocation(int x, int y, const FontGlyph* glyph) :
|
||||
GlyphLocation(float x, float y, const FontGlyph* glyph) :
|
||||
x_(x),
|
||||
y_(y),
|
||||
glyph_(glyph)
|
||||
@ -62,9 +62,9 @@ struct GlyphLocation
|
||||
}
|
||||
|
||||
/// X coordinate.
|
||||
int x_;
|
||||
float x_;
|
||||
/// Y coordinate.
|
||||
int y_;
|
||||
float y_;
|
||||
/// Glyph.
|
||||
const FontGlyph* glyph_;
|
||||
};
|
||||
@ -177,7 +177,7 @@ public:
|
||||
const Color& GetEffectColor() const { return effectColor_; }
|
||||
|
||||
/// Return row height.
|
||||
int GetRowHeight() const { return rowHeight_; }
|
||||
float GetRowHeight() const { return rowHeight_; }
|
||||
|
||||
/// Return number of rows.
|
||||
unsigned GetNumRows() const { return rowWidths_.Size(); }
|
||||
@ -186,11 +186,11 @@ public:
|
||||
unsigned GetNumChars() const { return unicodeText_.Size(); }
|
||||
|
||||
/// Return width of row by index.
|
||||
int GetRowWidth(unsigned index) const;
|
||||
float GetRowWidth(unsigned index) const;
|
||||
/// Return position of character by index relative to the text element origin.
|
||||
IntVector2 GetCharPosition(unsigned index);
|
||||
Vector2 GetCharPosition(unsigned index);
|
||||
/// Return size of character by index.
|
||||
IntVector2 GetCharSize(unsigned index);
|
||||
Vector2 GetCharSize(unsigned index);
|
||||
|
||||
/// Set text effect Z bias. Zero by default, adjusted only in 3D mode.
|
||||
void SetEffectDepthBias(float bias);
|
||||
@ -220,7 +220,7 @@ protected:
|
||||
int GetRowStartPosition(unsigned rowIndex) const;
|
||||
/// Contruct batch.
|
||||
void ConstructBatch
|
||||
(UIBatch& pageBatch, const PODVector<GlyphLocation>& pageGlyphLocation, int dx = 0, int dy = 0, Color* color = 0,
|
||||
(UIBatch& pageBatch, const PODVector<GlyphLocation>& pageGlyphLocation, float dx = 0, float dy = 0, Color* color = 0,
|
||||
float depthBias = 0.0f);
|
||||
|
||||
/// Font.
|
||||
@ -260,7 +260,7 @@ protected:
|
||||
/// Text effect Z bias.
|
||||
float effectDepthBias_;
|
||||
/// Row height.
|
||||
int rowHeight_;
|
||||
float rowHeight_;
|
||||
/// Text as Unicode characters.
|
||||
PODVector<unsigned> unicodeText_;
|
||||
/// Text modified into printed form.
|
||||
@ -268,7 +268,7 @@ protected:
|
||||
/// Mapping of printed form back to original char indices.
|
||||
PODVector<unsigned> printToText_;
|
||||
/// Row widths.
|
||||
PODVector<int> rowWidths_;
|
||||
PODVector<float> rowWidths_;
|
||||
/// Glyph locations per each texture in the font.
|
||||
Vector<PODVector<GlyphLocation> > pageGlyphLocations_;
|
||||
/// Cached locations of each character in the text.
|
||||
|
@ -487,12 +487,12 @@ int Text3D::GetRowWidth(unsigned index) const
|
||||
return text_.GetRowWidth(index);
|
||||
}
|
||||
|
||||
IntVector2 Text3D::GetCharPosition(unsigned index)
|
||||
Vector2 Text3D::GetCharPosition(unsigned index)
|
||||
{
|
||||
return text_.GetCharPosition(index);
|
||||
}
|
||||
|
||||
IntVector2 Text3D::GetCharSize(unsigned index)
|
||||
Vector2 Text3D::GetCharSize(unsigned index)
|
||||
{
|
||||
return text_.GetCharSize(index);
|
||||
}
|
||||
|
@ -144,9 +144,9 @@ public:
|
||||
/// Return width of row by index.
|
||||
int GetRowWidth(unsigned index) const;
|
||||
/// Return position of character by index relative to the text element origin.
|
||||
IntVector2 GetCharPosition(unsigned index);
|
||||
Vector2 GetCharPosition(unsigned index);
|
||||
/// Return size of character by index.
|
||||
IntVector2 GetCharSize(unsigned index);
|
||||
Vector2 GetCharSize(unsigned index);
|
||||
/// Return corner color.
|
||||
const Color& GetColor(Corner corner) const;
|
||||
/// Return opacity.
|
||||
|
@ -107,6 +107,7 @@ UI::UI(Context* context) :
|
||||
useMutableGlyphs_(false),
|
||||
forceAutoHint_(false),
|
||||
fontHintLevel_(FONT_HINT_LEVEL_NORMAL),
|
||||
subpixelGlyphPositions_(false),
|
||||
uiRendered_(false),
|
||||
nonModalBatchSize_(0),
|
||||
dragElementsCount_(0),
|
||||
@ -615,6 +616,15 @@ void UI::SetFontHintLevel(FontHintLevel level)
|
||||
}
|
||||
}
|
||||
|
||||
void UI::SetSubpixelGlyphPositions(bool enable)
|
||||
{
|
||||
if (enable != subpixelGlyphPositions_)
|
||||
{
|
||||
subpixelGlyphPositions_ = enable;
|
||||
ReleaseFontFaces();
|
||||
}
|
||||
}
|
||||
|
||||
void UI::SetScale(float scale)
|
||||
{
|
||||
uiScale_ = Max(scale, M_EPSILON);
|
||||
|
@ -110,6 +110,8 @@ public:
|
||||
void SetForceAutoHint(bool enable);
|
||||
/// Set the hinting level used by FreeType fonts.
|
||||
void SetFontHintLevel(FontHintLevel level);
|
||||
/// Set whether text glyphs can have fractional positions. Default is false (pixel-aligned).
|
||||
void SetSubpixelGlyphPositions(bool enable);
|
||||
/// Set %UI scale. 1.0 is default (pixel perfect). Resize the root element to match.
|
||||
void SetScale(float scale);
|
||||
/// Scale %UI to the specified width in pixels.
|
||||
@ -186,6 +188,9 @@ public:
|
||||
/// Return the current FreeType font hinting level.
|
||||
FontHintLevel GetFontHintLevel() const { return fontHintLevel_; }
|
||||
|
||||
// Return whether text glyphs can have fractional positions.
|
||||
bool GetSubpixelGlyphPositions() const { return subpixelGlyphPositions_; }
|
||||
|
||||
/// Return true when UI has modal element(s).
|
||||
bool HasModalElement() const;
|
||||
|
||||
@ -351,6 +356,8 @@ private:
|
||||
bool forceAutoHint_;
|
||||
/// FreeType hinting level (default is FONT_HINT_LEVEL_NORMAL).
|
||||
FontHintLevel fontHintLevel_;
|
||||
/// Flag for subpixel text glyph positions.
|
||||
bool subpixelGlyphPositions_;
|
||||
/// Flag for UI already being rendered this frame.
|
||||
bool uiRendered_;
|
||||
/// Non-modal batch size (used internally for rendering).
|
||||
|
@ -82,7 +82,7 @@ void UIBatch::SetDefaultColor()
|
||||
}
|
||||
}
|
||||
|
||||
void UIBatch::AddQuad(int x, int y, int width, int height, int texOffsetX, int texOffsetY, int texWidth, int texHeight)
|
||||
void UIBatch::AddQuad(float x, float y, float width, float height, int texOffsetX, int texOffsetY, int texWidth, int texHeight)
|
||||
{
|
||||
unsigned topLeftColor, topRightColor, bottomLeftColor, bottomRightColor;
|
||||
|
||||
@ -107,10 +107,10 @@ void UIBatch::AddQuad(int x, int y, int width, int height, int texOffsetX, int t
|
||||
|
||||
const IntVector2& screenPos = element_->GetScreenPosition();
|
||||
|
||||
float left = (float)(x + screenPos.x_) - posAdjust.x_;
|
||||
float right = left + (float)width;
|
||||
float top = (float)(y + screenPos.y_) - posAdjust.x_;
|
||||
float bottom = top + (float)height;
|
||||
float left = x + screenPos.x_ - posAdjust.x_;
|
||||
float right = left + width;
|
||||
float top = y + screenPos.y_ - posAdjust.x_;
|
||||
float bottom = top + height;
|
||||
|
||||
float leftUV = texOffsetX * invTextureSize_.x_;
|
||||
float topUV = texOffsetY * invTextureSize_.y_;
|
||||
@ -422,14 +422,14 @@ bool UIBatch::Merge(const UIBatch& batch)
|
||||
return true;
|
||||
}
|
||||
|
||||
unsigned UIBatch::GetInterpolatedColor(int x, int y)
|
||||
unsigned UIBatch::GetInterpolatedColor(float x, float y)
|
||||
{
|
||||
const IntVector2& size = element_->GetSize();
|
||||
|
||||
if (size.x_ && size.y_)
|
||||
{
|
||||
float cLerpX = Clamp((float)x / (float)size.x_, 0.0f, 1.0f);
|
||||
float cLerpY = Clamp((float)y / (float)size.y_, 0.0f, 1.0f);
|
||||
float cLerpX = Clamp(x / (float)size.x_, 0.0f, 1.0f);
|
||||
float cLerpY = Clamp(y / (float)size.y_, 0.0f, 1.0f);
|
||||
|
||||
Color topColor = element_->GetColor(C_TOPLEFT).Lerp(element_->GetColor(C_TOPRIGHT), cLerpX);
|
||||
Color bottomColor = element_->GetColor(C_BOTTOMLEFT).Lerp(element_->GetColor(C_BOTTOMRIGHT), cLerpX);
|
||||
|
@ -51,7 +51,7 @@ public:
|
||||
/// Restore UI element's default color.
|
||||
void SetDefaultColor();
|
||||
/// Add a quad.
|
||||
void AddQuad(int x, int y, int width, int height, int texOffsetX, int texOffsetY, int texWidth = 0, int texHeight = 0);
|
||||
void AddQuad(float x, float y, float width, float height, int texOffsetX, int texOffsetY, int texWidth = 0, int texHeight = 0);
|
||||
/// Add a quad using a transform matrix.
|
||||
void AddQuad(const Matrix3x4& transform, int x, int y, int width, int height, int texOffsetX, int texOffsetY, int texWidth = 0,
|
||||
int texHeight = 0);
|
||||
@ -67,7 +67,7 @@ public:
|
||||
/// Merge with another batch.
|
||||
bool Merge(const UIBatch& batch);
|
||||
/// Return an interpolated color for the UI element.
|
||||
unsigned GetInterpolatedColor(int x, int y);
|
||||
unsigned GetInterpolatedColor(float x, float y);
|
||||
|
||||
/// Add or merge a batch.
|
||||
static void AddOrMerge(const UIBatch& batch, PODVector<UIBatch>& batches);
|
||||
|
@ -36,10 +36,6 @@ function Start()
|
||||
CreateCheckbox("White background", "HandleWhiteBackground")
|
||||
:SetChecked(false)
|
||||
|
||||
-- Add a checkbox for the global ForceAutoHint setting. This affects character spacing.
|
||||
CreateCheckbox("UI::SetForceAutoHint", "HandleForceAutoHint")
|
||||
:SetChecked(ui:GetForceAutoHint())
|
||||
|
||||
-- Add a checkbox to toggle SRGB output conversion (if available).
|
||||
-- This will give more correct text output for FreeType fonts, as the FreeType rasterizer
|
||||
-- outputs linear coverage values rather than SRGB values. However, this feature isn't
|
||||
@ -47,6 +43,14 @@ function Start()
|
||||
CreateCheckbox("Graphics::SetSRGB", "HandleSRGB")
|
||||
:SetChecked(graphics:GetSRGB())
|
||||
|
||||
-- Add a checkbox for the global ForceAutoHint setting. This affects character spacing.
|
||||
CreateCheckbox("UI::SetForceAutoHint", "HandleForceAutoHint")
|
||||
:SetChecked(ui:GetForceAutoHint())
|
||||
|
||||
-- Add a checkbox for the global SubpixelGlyphPositions setting. This affects character spacing.
|
||||
CreateCheckbox("UI::SetSubpixelGlyphPositions", "HandleSubpixelGlyphPositions")
|
||||
:SetChecked(ui:GetSubpixelGlyphPositions())
|
||||
|
||||
-- Add a drop-down menu to control the font hinting level.
|
||||
local items = {
|
||||
"FONT_HINT_LEVEL_NONE",
|
||||
@ -161,6 +165,13 @@ function HandleSRGB(eventType, eventData)
|
||||
end
|
||||
end
|
||||
|
||||
function HandleSubpixelGlyphPositions(eventType, eventData)
|
||||
local box = eventData["Element"]:GetPtr("CheckBox")
|
||||
local checked = box:IsChecked()
|
||||
|
||||
ui:SetSubpixelGlyphPositions(checked)
|
||||
end
|
||||
|
||||
function HandleFontHintLevel(eventType, eventData)
|
||||
local list = eventData["Element"]:GetPtr("DropDownList")
|
||||
local i = list:GetSelection()
|
||||
|
@ -37,10 +37,6 @@ void Start()
|
||||
CreateCheckbox("White background", "HandleWhiteBackground")
|
||||
.checked = false;
|
||||
|
||||
// Add a checkbox for the global ForceAutoHint setting. This affects character spacing.
|
||||
CreateCheckbox("UI::SetForceAutoHint", "HandleForceAutoHint")
|
||||
.checked = ui.forceAutoHint;
|
||||
|
||||
// Add a checkbox to toggle SRGB output conversion (if available).
|
||||
// This will give more correct text output for FreeType fonts, as the FreeType rasterizer
|
||||
// outputs linear coverage values rather than SRGB values. However, this feature isn't
|
||||
@ -48,6 +44,14 @@ void Start()
|
||||
CreateCheckbox("Graphics::SetSRGB", "HandleSRGB")
|
||||
.checked = graphics.sRGB;
|
||||
|
||||
// Add a checkbox for the global ForceAutoHint setting. This affects character spacing.
|
||||
CreateCheckbox("UI::SetForceAutoHint", "HandleForceAutoHint")
|
||||
.checked = ui.forceAutoHint;
|
||||
|
||||
// Add a checkbox for the global SubpixelGlyphPositions setting. This affects character spacing.
|
||||
CreateCheckbox("UI::SetSubpixelGlyphPositions", "HandleSubpixelGlyphPositions")
|
||||
.checked = ui.subpixelGlyphPositions;
|
||||
|
||||
// Add a drop-down menu to control the font hinting level.
|
||||
Array<String> items = {
|
||||
"FONT_HINT_LEVEL_NONE",
|
||||
@ -174,6 +178,15 @@ void HandleSRGB(StringHash eventType, VariantMap& eventData)
|
||||
}
|
||||
}
|
||||
|
||||
void HandleSubpixelGlyphPositions(StringHash eventType, VariantMap& eventData)
|
||||
{
|
||||
CheckBox@ box = eventData["Element"].GetPtr();
|
||||
bool checked = box.checked;
|
||||
|
||||
ui.subpixelGlyphPositions = checked;
|
||||
}
|
||||
|
||||
|
||||
void HandleFontHintLevel(StringHash eventType, VariantMap& eventData)
|
||||
{
|
||||
DropDownList@ list = eventData["Element"].GetPtr();
|
||||
|
Loading…
Reference in New Issue
Block a user