2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2008 Dolphin Emulator Project
|
2015-05-18 01:08:10 +02:00
|
|
|
// Licensed under GPLv2+
|
2013-04-18 05:09:55 +02:00
|
|
|
// Refer to the license.txt file included.
|
2008-12-08 05:46:09 +01:00
|
|
|
|
2014-02-10 19:54:46 +01:00
|
|
|
#pragma once
|
2008-12-08 05:46:09 +01:00
|
|
|
|
2014-02-17 11:18:15 +01:00
|
|
|
#include <cstdarg>
|
2014-02-20 04:11:52 +01:00
|
|
|
#include <cstddef>
|
2019-12-31 00:28:35 +01:00
|
|
|
#include <cstdlib>
|
2010-11-10 05:12:31 +01:00
|
|
|
#include <iomanip>
|
2020-03-05 22:00:28 +01:00
|
|
|
#include <limits>
|
2019-12-31 00:28:35 +01:00
|
|
|
#include <locale>
|
2014-02-17 11:18:15 +01:00
|
|
|
#include <sstream>
|
|
|
|
#include <string>
|
2018-06-03 14:10:52 +02:00
|
|
|
#include <type_traits>
|
2014-02-17 11:18:15 +01:00
|
|
|
#include <vector>
|
2008-12-08 05:46:09 +01:00
|
|
|
|
2015-09-26 23:13:07 +02:00
|
|
|
#include "Common/CommonTypes.h"
|
2008-12-08 05:46:09 +01:00
|
|
|
|
2019-06-21 18:07:59 +02:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
#include <filesystem>
|
|
|
|
#define HAS_STD_FILESYSTEM
|
|
|
|
#endif
|
|
|
|
|
2014-12-29 01:09:07 +01:00
|
|
|
std::string StringFromFormatV(const char* format, va_list args);
|
|
|
|
|
2014-02-06 13:10:04 +01:00
|
|
|
std::string StringFromFormat(const char* format, ...)
|
|
|
|
#if !defined _WIN32
|
|
|
|
// On compilers that support function attributes, this gives StringFromFormat
|
|
|
|
// the same errors and warnings that printf would give.
|
|
|
|
__attribute__((__format__(printf, 1, 2)))
|
|
|
|
#endif
|
|
|
|
;
|
|
|
|
|
2008-12-08 05:46:09 +01:00
|
|
|
// Cheap!
|
|
|
|
bool CharArrayFromFormatV(char* out, int outsize, const char* format, va_list args);
|
|
|
|
|
|
|
|
template <size_t Count>
|
|
|
|
inline void CharArrayFromFormat(char (&out)[Count], const char* format, ...)
|
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
va_start(args, format);
|
|
|
|
CharArrayFromFormatV(out, Count, format, args);
|
|
|
|
va_end(args);
|
|
|
|
}
|
|
|
|
|
2010-11-10 05:12:31 +01:00
|
|
|
// Good
|
2016-01-21 21:16:51 +01:00
|
|
|
std::string ArrayToString(const u8* data, u32 size, int line_len = 20, bool spaces = true);
|
2010-11-10 05:12:31 +01:00
|
|
|
|
2019-07-08 13:35:53 +02:00
|
|
|
std::string_view StripSpaces(std::string_view s);
|
|
|
|
std::string_view StripQuotes(std::string_view s);
|
2010-11-10 05:12:31 +01:00
|
|
|
|
2019-12-31 00:28:35 +01:00
|
|
|
std::string ReplaceAll(std::string result, std::string_view src, std::string_view dest);
|
|
|
|
|
2016-01-21 21:27:56 +01:00
|
|
|
bool TryParse(const std::string& str, bool* output);
|
2008-12-08 05:46:09 +01:00
|
|
|
|
2019-12-31 00:28:35 +01:00
|
|
|
template <typename T, std::enable_if_t<std::is_integral_v<T> || std::is_enum_v<T>>* = nullptr>
|
2020-05-24 01:40:10 +02:00
|
|
|
bool TryParse(const std::string& str, T* output, int base = 0)
|
2010-11-10 05:12:31 +01:00
|
|
|
{
|
2019-12-31 00:28:35 +01:00
|
|
|
char* end_ptr = nullptr;
|
|
|
|
|
|
|
|
// Set errno to a clean slate.
|
|
|
|
errno = 0;
|
|
|
|
|
|
|
|
// Read a u64 for unsigned types and s64 otherwise.
|
|
|
|
using ReadType = std::conditional_t<std::is_unsigned_v<T>, u64, s64>;
|
|
|
|
ReadType value;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-12-31 00:28:35 +01:00
|
|
|
if constexpr (std::is_unsigned_v<T>)
|
2020-05-24 01:40:10 +02:00
|
|
|
value = std::strtoull(str.c_str(), &end_ptr, base);
|
2019-12-31 00:28:35 +01:00
|
|
|
else
|
2020-05-24 01:40:10 +02:00
|
|
|
value = std::strtoll(str.c_str(), &end_ptr, base);
|
2019-12-31 00:28:35 +01:00
|
|
|
|
|
|
|
// Fail if the end of the string wasn't reached.
|
|
|
|
if (end_ptr == nullptr || *end_ptr != '\0')
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Fail if the value was out of 64-bit range.
|
|
|
|
if (errno == ERANGE)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
using LimitsType = typename std::conditional_t<std::is_enum_v<T>, std::underlying_type<T>,
|
|
|
|
std::common_type<T>>::type;
|
|
|
|
// Fail if outside numeric limits.
|
|
|
|
if (value < std::numeric_limits<LimitsType>::min() ||
|
|
|
|
value > std::numeric_limits<LimitsType>::max())
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
*output = static_cast<T>(value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, std::enable_if_t<std::is_floating_point_v<T>>* = nullptr>
|
|
|
|
bool TryParse(std::string str, T* const output)
|
|
|
|
{
|
|
|
|
// Replace commas with dots.
|
|
|
|
std::istringstream iss(ReplaceAll(std::move(str), ",", "."));
|
|
|
|
|
|
|
|
// Use "classic" locale to force a "dot" decimal separator.
|
|
|
|
iss.imbue(std::locale::classic());
|
|
|
|
|
|
|
|
T tmp;
|
|
|
|
|
|
|
|
// Succeed if a value was read and the entire string was used.
|
2019-10-19 02:56:48 +02:00
|
|
|
if (iss >> tmp && iss.eof())
|
2010-11-10 05:12:31 +01:00
|
|
|
{
|
|
|
|
*output = tmp;
|
|
|
|
return true;
|
|
|
|
}
|
2018-06-15 14:11:18 +02:00
|
|
|
|
|
|
|
return false;
|
2010-11-10 05:12:31 +01:00
|
|
|
}
|
2008-12-08 05:46:09 +01:00
|
|
|
|
2014-07-29 18:34:57 +02:00
|
|
|
template <typename N>
|
|
|
|
bool TryParseVector(const std::string& str, std::vector<N>* output, const char delimiter = ',')
|
|
|
|
{
|
|
|
|
output->clear();
|
|
|
|
std::istringstream buffer(str);
|
|
|
|
std::string variable;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2014-07-29 18:34:57 +02:00
|
|
|
while (std::getline(buffer, variable, delimiter))
|
|
|
|
{
|
|
|
|
N tmp = 0;
|
|
|
|
if (!TryParse(variable, &tmp))
|
|
|
|
return false;
|
|
|
|
output->push_back(tmp);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-06-03 14:10:52 +02:00
|
|
|
std::string ValueToString(u16 value);
|
|
|
|
std::string ValueToString(u32 value);
|
|
|
|
std::string ValueToString(u64 value);
|
|
|
|
std::string ValueToString(float value);
|
|
|
|
std::string ValueToString(double value);
|
|
|
|
std::string ValueToString(int value);
|
|
|
|
std::string ValueToString(s64 value);
|
|
|
|
std::string ValueToString(bool value);
|
|
|
|
template <typename T, std::enable_if_t<std::is_enum<T>::value>* = nullptr>
|
|
|
|
std::string ValueToString(T value)
|
|
|
|
{
|
|
|
|
return ValueToString(static_cast<std::underlying_type_t<T>>(value));
|
|
|
|
}
|
|
|
|
|
2016-06-17 02:08:12 +02:00
|
|
|
// Generates an hexdump-like representation of a binary data blob.
|
|
|
|
std::string HexDump(const u8* data, size_t size);
|
|
|
|
|
2008-12-08 05:46:09 +01:00
|
|
|
// TODO: kill this
|
2014-03-12 20:33:41 +01:00
|
|
|
bool AsciiToHex(const std::string& _szValue, u32& result);
|
2008-12-08 05:46:09 +01:00
|
|
|
|
2019-07-08 13:35:53 +02:00
|
|
|
std::string TabsToSpaces(int tab_size, std::string str);
|
2009-06-21 10:39:21 +02:00
|
|
|
|
2017-06-11 16:33:10 +02:00
|
|
|
std::vector<std::string> SplitString(const std::string& str, char delim);
|
2016-11-26 15:39:00 +01:00
|
|
|
std::string JoinStrings(const std::vector<std::string>& strings, const std::string& delimiter);
|
2008-12-08 05:46:09 +01:00
|
|
|
|
2010-11-10 05:12:31 +01:00
|
|
|
// "C:/Windows/winhelp.exe" to "C:/Windows/", "winhelp", ".exe"
|
2019-07-08 21:08:35 +02:00
|
|
|
bool SplitPath(std::string_view full_path, std::string* path, std::string* filename,
|
|
|
|
std::string* extension);
|
2009-09-04 13:48:49 +02:00
|
|
|
|
2020-03-16 21:03:21 +01:00
|
|
|
std::string PathToFileName(std::string_view path);
|
|
|
|
|
2019-07-08 21:08:35 +02:00
|
|
|
void BuildCompleteFilename(std::string& complete_filename, std::string_view path,
|
|
|
|
std::string_view filename);
|
2008-12-08 05:46:09 +01:00
|
|
|
|
2019-07-08 13:35:53 +02:00
|
|
|
bool StringBeginsWith(std::string_view str, std::string_view begin);
|
|
|
|
bool StringEndsWith(std::string_view str, std::string_view end);
|
2017-06-06 06:08:51 +02:00
|
|
|
void StringPopBackIf(std::string* s, char c);
|
2018-11-21 06:15:44 +01:00
|
|
|
size_t StringUTF8CodePointCount(const std::string& str);
|
2016-12-06 16:43:41 +01:00
|
|
|
|
2019-07-08 13:35:53 +02:00
|
|
|
std::string CP1252ToUTF8(std::string_view str);
|
|
|
|
std::string SHIFTJISToUTF8(std::string_view str);
|
|
|
|
std::string UTF8ToSHIFTJIS(std::string_view str);
|
2020-06-28 18:15:30 +02:00
|
|
|
std::string WStringToUTF8(std::wstring_view str);
|
2017-11-02 17:05:45 +01:00
|
|
|
std::string UTF16BEToUTF8(const char16_t* str, size_t max_size); // Stops at \0
|
2020-06-28 19:16:23 +02:00
|
|
|
std::string UTF16ToUTF8(std::u16string_view str);
|
|
|
|
std::u16string UTF8ToUTF16(std::string_view str);
|
2013-03-03 02:46:55 +01:00
|
|
|
|
2013-02-28 01:00:42 +01:00
|
|
|
#ifdef _WIN32
|
|
|
|
|
2020-06-28 18:15:30 +02:00
|
|
|
std::wstring UTF8ToWString(std::string_view str);
|
2013-02-28 01:00:42 +01:00
|
|
|
|
2013-02-28 01:51:02 +01:00
|
|
|
#ifdef _UNICODE
|
2019-07-08 13:35:53 +02:00
|
|
|
inline std::string TStrToUTF8(std::wstring_view str)
|
2013-02-28 01:51:02 +01:00
|
|
|
{
|
2020-06-28 18:15:30 +02:00
|
|
|
return WStringToUTF8(str);
|
2013-02-28 01:51:02 +01:00
|
|
|
}
|
|
|
|
|
2019-07-08 13:35:53 +02:00
|
|
|
inline std::wstring UTF8ToTStr(std::string_view str)
|
2013-02-28 01:51:02 +01:00
|
|
|
{
|
2020-06-28 18:15:30 +02:00
|
|
|
return UTF8ToWString(str);
|
2013-02-28 01:51:02 +01:00
|
|
|
}
|
|
|
|
#else
|
2019-07-08 13:35:53 +02:00
|
|
|
inline std::string TStrToUTF8(std::string_view str)
|
2013-02-28 01:51:02 +01:00
|
|
|
{
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2019-07-08 13:35:53 +02:00
|
|
|
inline std::string UTF8ToTStr(std::string_view str)
|
2013-02-28 01:51:02 +01:00
|
|
|
{
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2013-02-28 01:00:42 +01:00
|
|
|
#endif
|
2018-05-08 13:55:07 +02:00
|
|
|
|
2019-06-21 18:07:59 +02:00
|
|
|
#ifdef HAS_STD_FILESYSTEM
|
|
|
|
std::filesystem::path StringToPath(std::string_view path);
|
|
|
|
std::string PathToString(const std::filesystem::path& path);
|
|
|
|
#endif
|
|
|
|
|
2018-05-08 13:55:07 +02:00
|
|
|
// Thousand separator. Turns 12345678 into 12,345,678
|
|
|
|
template <typename I>
|
|
|
|
std::string ThousandSeparate(I value, int spaces = 0)
|
|
|
|
{
|
|
|
|
#ifdef _WIN32
|
|
|
|
std::wostringstream stream;
|
|
|
|
#else
|
|
|
|
std::ostringstream stream;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
stream << std::setw(spaces) << value;
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
2020-06-28 18:15:30 +02:00
|
|
|
return WStringToUTF8(stream.str());
|
2018-05-08 13:55:07 +02:00
|
|
|
#else
|
|
|
|
return stream.str();
|
|
|
|
#endif
|
|
|
|
}
|
2019-12-30 10:48:11 +01:00
|
|
|
|
|
|
|
/// Returns whether a character is printable, i.e. whether 0x20 <= c <= 0x7e is true.
|
|
|
|
/// Use this instead of calling std::isprint directly to ensure
|
|
|
|
/// the C locale is being used and to avoid possibly undefined behaviour.
|
|
|
|
inline bool IsPrintableCharacter(char c)
|
|
|
|
{
|
|
|
|
return std::isprint(c, std::locale::classic());
|
|
|
|
}
|