mirror of
				https://github.com/ciphervance/supercell-wx.git
				synced 2025-11-03 23:50:05 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			176 lines
		
	
	
	
		
			3.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			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
 |