supercell-wx/wxdata/source/scwx/util/strings.cpp
2024-03-21 22:21:03 -05:00

176 lines
3.9 KiB
C++

#define STRINGS_IMPLEMENTATION
#include <scwx/util/strings.hpp>
#include <boost/algorithm/string/trim.hpp>
#include <boost/lexical_cast.hpp>
#include <fmt/format.h>
namespace scwx
{
namespace util
{
std::string BytesToString(std::ptrdiff_t bytes)
{
auto FormatNumber = [](double number) -> std::string
{
int precision;
// Determine precision
if (number >= 100.0)
{
precision = 0;
}
else if (number >= 10.0)
{
precision = 1;
}
else
{
precision = 2;
}
// Format the number
std::string formattedNum = fmt::format("{:.{}f}", number, precision);
// Remove trailing zeroes
std::size_t found = formattedNum.find_last_not_of('0');
if (found != std::string::npos && formattedNum[found] == '.')
{
// Keep one trailing zero if it's a decimal point
found++;
}
formattedNum.erase(found + 1, std::string::npos);
return formattedNum;
};
// Print with appropriate suffix
if (bytes < 1000)
{
return fmt::format("{} bytes", bytes);
}
double kilobytes = bytes / 1024.0;
if (kilobytes < 1000.0)
{
return fmt::format("{} KB", FormatNumber(kilobytes));
}
double megabytes = kilobytes / 1024.0;
if (megabytes < 1000.0)
{
return fmt::format("{} MB", FormatNumber(megabytes));
}
double gigabytes = megabytes / 1024.0;
if (gigabytes < 1000.0)
{
return fmt::format("{} GB", FormatNumber(gigabytes));
}
double terabytes = gigabytes / 1024.0;
return fmt::format("{} TB", FormatNumber(terabytes));
}
std::vector<std::string> ParseTokens(const std::string& s,
std::vector<std::string> delimiters,
std::size_t pos)
{
std::vector<std::string> tokens {};
std::size_t findPos {};
// Iterate through each delimiter
for (std::size_t i = 0; i < delimiters.size() && pos != std::string::npos;
++i)
{
// Skip leading spaces
while (pos < s.size() && std::isspace(s[pos]))
{
++pos;
}
if (pos < s.size() && s[pos] == '"')
{
// Do not search for a delimeter within a quoted string
findPos = s.find('"', pos + 1);
// Increment search start to one after quotation mark
if (findPos != std::string::npos)
{
++findPos;
}
}
else
{
// Search starting at the current position
findPos = pos;
}
// Search for delimiter
std::size_t nextPos = s.find_first_of(delimiters[i], findPos);
// If the delimiter was not found, stop processing tokens
if (nextPos == std::string::npos)
{
break;
}
// Add the current substring as a token
auto& newToken = tokens.emplace_back(s.substr(pos, nextPos - pos));
boost::trim(newToken);
// Increment nextPos until the next non-space character
while (++nextPos < s.size() && std::isspace(s[nextPos])) {}
// Store new position value
pos = nextPos;
}
// Add the remainder of the string as a token
if (pos < s.size())
{
auto& newToken = tokens.emplace_back(s.substr(pos));
boost::trim(newToken);
}
return tokens;
}
std::string ToString(const std::vector<std::string>& v)
{
std::string value {};
for (const std::string& s : v)
{
if (!value.empty())
{
value += ", ";
}
value += s;
}
return value;
}
template<typename T>
std::optional<T> TryParseNumeric(const std::string& str)
{
std::optional<T> value = std::nullopt;
try
{
auto trimmed = boost::algorithm::trim_copy(str);
value = boost::lexical_cast<T>(trimmed);
}
catch (const std::exception&)
{
}
return value;
}
} // namespace util
} // namespace scwx