mirror of
https://github.com/4jcraft/4jcraft.git
synced 2026-04-23 22:13:37 +00:00
179 lines
6 KiB
C++
179 lines
6 KiB
C++
#include "../Platform/stdafx.h"
|
|
#include "simdutf.h"
|
|
|
|
std::wstring toLower(const std::wstring& a) {
|
|
std::wstring out = std::wstring(a);
|
|
std::transform(out.begin(), out.end(), out.begin(), ::tolower);
|
|
return out;
|
|
}
|
|
|
|
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<const wchar_t*>(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<char32_t*>(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<const char16_t*>(converting.data()),
|
|
converting.size());
|
|
} else if constexpr (sizeof(wchar_t) == 4) {
|
|
// POSIX, UTF-32
|
|
if (converting.empty()) return {};
|
|
|
|
const char32_t* data32 =
|
|
reinterpret_cast<const char32_t*>(converting.data());
|
|
const std::size_t len32 = converting.size();
|
|
|
|
std::u16string result(simdutf::utf16_length_from_utf32(data32, len32),
|
|
u'\0');
|
|
std::size_t convertedLength =
|
|
simdutf::convert_utf32_to_utf16(data32, len32, 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.");
|
|
}
|
|
}
|
|
|
|
// Convert for filename std::wstrings to a straight character pointer for Xbox
|
|
// APIs. The returned string is only valid until this function is called again,
|
|
// and it isn't thread-safe etc. as I'm just storing the returned name in a
|
|
// local static to save having to clear it up everywhere this is used.
|
|
const char* wstringtofilename(const std::wstring& name) {
|
|
static char buf[256];
|
|
assert(name.length() < 256);
|
|
for (unsigned int i = 0; i < name.length(); i++) {
|
|
wchar_t c = name[i];
|
|
#if defined __PS3__ || defined __ORBIS__ || defined __linux__
|
|
if (c == '\\') c = '/';
|
|
#else
|
|
if (c == '/') c = '\\';
|
|
#endif
|
|
assert(c < 128); // Will we have to do any conversion of non-ASCII
|
|
// characters in filenames?
|
|
buf[i] = (char)c;
|
|
}
|
|
buf[name.length()] = 0;
|
|
return buf;
|
|
}
|
|
|
|
std::wstring filenametowstring(const char* name) {
|
|
return convStringToWstring(name);
|
|
}
|
|
|
|
std::vector<std::wstring>& stringSplit(const std::wstring& s, wchar_t delim,
|
|
std::vector<std::wstring>& elems) {
|
|
std::wstringstream ss(s);
|
|
std::wstring item;
|
|
while (std::getline(ss, item, delim)) {
|
|
elems.push_back(item);
|
|
}
|
|
return elems;
|
|
}
|
|
|
|
std::vector<std::wstring> stringSplit(const std::wstring& s, wchar_t delim) {
|
|
std::vector<std::wstring> 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;
|
|
}
|