#include "CTextParser.hpp" #include "CFontImageDef.hpp" #include "CTextExecuteBuffer.hpp" namespace urde { CTextColor CTextParser::ParseColor(const wchar_t* str, int len) { u8 r = GetColorValue(str + 1); u8 g = GetColorValue(str + 3); u8 b = GetColorValue(str + 5); u8 a = 0xff; if (len == 9) a = GetColorValue(str + 7); return CTextColor(r, g, b, a); } u8 CTextParser::GetColorValue(const wchar_t* str) { return (FromHex(str[0]) << 4) + FromHex(str[1]); } u32 CTextParser::FromHex(wchar_t ch) { if (ch >= L'0' && ch <= L'9') return ch - L'0'; if (ch >= L'A' && ch <= L'F') return ch - L'A' + 10; if (ch >= L'a' && ch <= L'f') return ch - L'a' + 10; return 0; } s32 CTextParser::ParseInt(const wchar_t* str, int len, bool signVal) { bool neg = false; int procCur = 0; if (signVal && len && *str == L'-') { neg = true; procCur = 1; } int val = 0; while (len > procCur) { val *= 10; wchar_t ch = str[procCur]; val += ch - L'0'; ++procCur; } return neg ? -val : val; } bool CTextParser::Equals(const wchar_t* str, int len, const wchar_t* other) { int i=0; for (; *other && i= L'0' && str[7] <= L'9') { ++valCur; --len; val *= 10; val += str[7] - L'0'; } if (Equals(valCur + 10, len - 10, L"no")) out.AddRemoveColorOverride(val); else { CTextColor color = ParseColor(str + 10, len - 10); out.AddColorOverride(val, color); } } else if (BeginsWith(str, len, L"line-spacing=")) { out.AddLineSpacing(ParseInt(str + 13, len - 13, true) / 100.0); } else if (BeginsWith(str, len, L"line-extra-space=")) { out.AddLineSpacing(ParseInt(str + 17, len - 17, true)); } else if (BeginsWith(str, len, L"just=")) { if (Equals(str + 5, len - 5, L"left")) out.AddJustification(EJustification::Left); else if (Equals(str + 5, len - 5, L"center")) out.AddJustification(EJustification::Center); else if (Equals(str + 5, len - 5, L"right")) out.AddJustification(EJustification::Right); else if (Equals(str + 5, len - 5, L"full")) out.AddJustification(EJustification::Full); else if (Equals(str + 5, len - 5, L"nleft")) out.AddJustification(EJustification::NLeft); else if (Equals(str + 5, len - 5, L"ncenter")) out.AddJustification(EJustification::NCenter); else if (Equals(str + 5, len - 5, L"nright")) out.AddJustification(EJustification::NRight); } else if (BeginsWith(str, len, L"vjust=")) { if (Equals(str + 6, len - 6, L"top")) out.AddVerticalJustification(EVerticalJustification::Top); else if (Equals(str + 6, len - 6, L"center")) out.AddVerticalJustification(EVerticalJustification::Center); else if (Equals(str + 6, len - 6, L"bottom")) out.AddVerticalJustification(EVerticalJustification::Bottom); else if (Equals(str + 6, len - 6, L"full")) out.AddVerticalJustification(EVerticalJustification::Full); else if (Equals(str + 6, len - 6, L"ntop")) out.AddVerticalJustification(EVerticalJustification::NTop); else if (Equals(str + 6, len - 6, L"ncenter")) out.AddVerticalJustification(EVerticalJustification::NCenter); else if (Equals(str + 6, len - 6, L"nbottom")) out.AddVerticalJustification(EVerticalJustification::NBottom); } else if (Equals(str, len, L"push")) { out.AddPushState(); } else if (Equals(str, len, L"pop")) { out.AddPopState(); } } CFontImageDef CTextParser::GetImage(const wchar_t* str, int len) { int commaCount = 0; for (int i=0 ; i> texs; texs.reserve(commaCount - 1); do { AdvanceCommaPos(); texs.push_back(x0_store.GetObj({SBIG('TXTR'), GetAssetIdFromString(&iterable[tokenPos])})); AdvanceTokenPos(); } while (commaPos != iterable.size()); return CFontImageDef(std::move(texs), interval, zeus::CVector2f(1.f, 1.f)); } else if (BeginsWith(str, len, L"SA")) { /* Scaled and animated texture array */ AdvanceCommaPos(); float interval = std::wcstof(&iterable[tokenPos], nullptr); AdvanceTokenPos(); AdvanceCommaPos(); float scaleX = std::wcstof(&iterable[tokenPos], nullptr); AdvanceTokenPos(); AdvanceCommaPos(); float scaleY = std::wcstof(&iterable[tokenPos], nullptr); AdvanceTokenPos(); std::vector> texs; texs.reserve(commaCount - 3); do { AdvanceCommaPos(); texs.push_back(x0_store.GetObj({SBIG('TXTR'), GetAssetIdFromString(&iterable[tokenPos])})); AdvanceTokenPos(); } while (commaPos != iterable.size()); return CFontImageDef(std::move(texs), interval, zeus::CVector2f(scaleX, scaleY)); } else if (BeginsWith(str, len, L"SI")) { /* Scaled single texture */ AdvanceCommaPos(); float scaleX = std::wcstof(&iterable[tokenPos], nullptr); AdvanceTokenPos(); AdvanceCommaPos(); float scaleY = std::wcstof(&iterable[tokenPos], nullptr); AdvanceTokenPos(); AdvanceCommaPos(); TToken tex = x0_store.GetObj({SBIG('TXTR'), GetAssetIdFromString(&iterable[tokenPos])}); AdvanceTokenPos(); return CFontImageDef(std::move(tex), zeus::CVector2f(scaleX, scaleY)); } } TToken tex = x0_store.GetObj({SBIG('TXTR'), GetAssetIdFromString(str)}); return CFontImageDef(std::move(tex), zeus::CVector2f(1.f, 1.f)); } TResId CTextParser::GetAssetIdFromString(const wchar_t* str) { u8 r = GetColorValue(str); u8 g = GetColorValue(str + 2); u8 b = GetColorValue(str + 4); u8 a = GetColorValue(str + 6); return (r << 24) | (g << 16) | (b << 8) | a; } TToken CTextParser::GetFont(const wchar_t* str, int len) { return x0_store.GetObj({SBIG('FONT'), GetAssetIdFromString(str)}); } void CTextParser::ParseText(CTextExecuteBuffer& out, const wchar_t* str, int len) { for (int b=0, e=0 ; str[e] && (len == -1 || e < len) ;) { if (str[e] != L'&') { ++e; continue; } if ((len == -1 || e+1 < len) && str[e+1] != L'&') { out.AddString(str + b, e - b); ++e; b = e; while (str[e] && (len == -1 || e < len) && str[e] != L';') ++e; ParseTag(out, str + e, e - b); b = e + 1; } else { out.AddString(str + b, e + 1 - b); e += 2; b = e; } } } }