#include #include #include #include #include #include #include #include "simdutf.h" std::wstring toLower(const std::wstring& a) { std::wstring out = std::wstring(a); std::transform(out.begin(), out.end(), out.begin(), std::towlower); return out; } // 4jcraft TODO: this intentionally returns the original string (not empty) // for whitespace-only input. Callers in animation file parsing // (AbstractTexturePack::getAnimationString) depend on this behavior - // returning empty here breaks clock/compass texture frame loading. std::wstring trimString(const std::wstring& a) { std::wstring b; int start = (int)a.find_first_not_of(L" \t\n\r"); int end = (int)a.find_last_not_of(L" \t\n\r"); if (start == std::wstring::npos) start = 0; if (end == std::wstring::npos) end = (int)a.size() - 1; b = a.substr(start, (end - start) + 1); return b; } std::wstring replaceAll(const std::wstring& in, const std::wstring& replace, const std::wstring& with) { std::wstring out = in; size_t pos = 0; while ((pos = out.find(replace, pos)) != std::wstring::npos) { out.replace(pos, replace.length(), with); pos++; } return out; } bool equalsIgnoreCase(const std::wstring& a, const std::wstring& b) { bool out; std::wstring c = toLower(a); std::wstring d = toLower(b); out = c.compare(d) == 0; return out; } std::wstring convStringToWstring(const std::string& converting) { std::wstring converted(converting.length(), L' '); copy(converting.begin(), converting.end(), converted.begin()); return converted; } std::wstring u16string_to_wstring(const std::u16string& converting) { if constexpr (sizeof(wchar_t) == 2) { // on Windows, wchar_t is UTF-16 so we can get away with just a type // transmutation return std::wstring(reinterpret_cast(converting.data()), converting.size()); } else if constexpr (sizeof(wchar_t) == 4) { // POSIX has wchar_t as UTF-32 instead so simdutf time :>>> if (converting.empty()) return {}; std::wstring result(simdutf::utf32_length_from_utf16(converting.data(), converting.size()), L'\0'); std::size_t convertedLength = simdutf::convert_utf16_to_utf32( converting.data(), converting.size(), reinterpret_cast(result.data())); result.resize(convertedLength); return result; } else { static_assert(sizeof(wchar_t) == 2 || sizeof(wchar_t) == 4, "Here's a nickel, Kid. Go buy yourself a real computer."); } } std::u16string wstring_to_u16string(const std::wstring& converting) { if constexpr (sizeof(wchar_t) == 2) { // Windows, UTF-16 return std::u16string( reinterpret_cast(converting.data()), converting.size()); } else if constexpr (sizeof(wchar_t) == 4) { // POSIX, UTF-32 if (converting.empty()) return {}; auto data32 = reinterpret_cast(converting.data()); auto len32 = converting.size(); std::u16string result(simdutf::utf16_length_from_utf32(data32, len32), u'\0'); auto len = simdutf::convert_utf32_to_utf16(data32, len32, result.data()); result.resize(len); return result; } else { static_assert(sizeof(wchar_t) == 2 || sizeof(wchar_t) == 4, "Here's a nickel, Kid. Go buy yourself a real computer."); } } std::u8string wstring_to_u8string(const std::wstring& converting) { if (converting.empty()) return {}; if constexpr (sizeof(wchar_t) == 2) { auto data16 = reinterpret_cast(converting.data()); auto len16 = converting.size(); std::u8string result(simdutf::utf8_length_from_utf16le(data16, len16), u'\0'); auto len = simdutf::convert_utf16_to_utf8(data16, len16, reinterpret_cast(result.data())); result.resize(len); return result; } else if constexpr (sizeof(wchar_t) == 4) { auto data32 = reinterpret_cast(converting.data()); auto len32 = converting.size(); std::u8string result(simdutf::utf8_length_from_utf32(data32, len32), u'\0'); auto len = simdutf::convert_utf32_to_utf8(data32, len32, reinterpret_cast(result.data())); result.resize(len); return result; } else { static_assert(sizeof(wchar_t) == 2 || sizeof(wchar_t) == 4, "Here's a nickel, Kid. Go buy yourself a real computer."); } } std::string wstringtofilename(const std::wstring& name) { std::string result; result.reserve(name.size()); for (wchar_t c : name) { #if defined(__linux__) if (c == L'\\') c = L'/'; #else if (c == L'/') c = L'\\'; #endif result += static_cast(c); } return result; } std::wstring filenametowstring(const char* name) { return convStringToWstring(name); } std::vector& stringSplit(const std::wstring& s, wchar_t delim, std::vector& elems) { std::wstringstream ss(s); std::wstring item; while (std::getline(ss, item, delim)) { elems.push_back(item); } return elems; } std::vector stringSplit(const std::wstring& s, wchar_t delim) { std::vector elems; return stringSplit(s, delim, elems); } bool BothAreSpaces(wchar_t lhs, wchar_t rhs) { return (lhs == rhs) && (lhs == L' '); } void stripWhitespaceForHtml(std::wstring& string, bool bRemoveNewline) { // Strip newline chars if (bRemoveNewline) { string.erase(std::remove(string.begin(), string.end(), '\n'), string.end()); string.erase(std::remove(string.begin(), string.end(), '\r'), string.end()); } string.erase(std::remove(string.begin(), string.end(), '\t'), string.end()); // Strip duplicate spaces string.erase(std::unique(string.begin(), string.end(), BothAreSpaces), string.end()); string = trimString(string); } std::wstring escapeXML(const std::wstring& in) { std::wstring out = in; out = replaceAll(out, L"&", L"&"); // out = replaceAll(out, L"\"", L"""); // out = replaceAll(out, L"'", L"'"); out = replaceAll(out, L"<", L"<"); out = replaceAll(out, L">", L">"); return out; } std::wstring parseXMLSpecials(const std::wstring& in) { std::wstring out = in; out = replaceAll(out, L"&", L"&"); // out = replaceAll(out, L"\"", L"""); // out = replaceAll(out, L"'", L"'"); out = replaceAll(out, L"<", L"<"); out = replaceAll(out, L">", L">"); return out; }