2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2008 Dolphin Emulator Project
|
2021-07-05 03:22:19 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2014-02-10 13:54:46 -05:00
|
|
|
#pragma once
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2023-06-10 07:56:40 -05:00
|
|
|
#include <charconv>
|
2014-02-17 05:18:15 -05:00
|
|
|
#include <cstdarg>
|
2014-02-20 04:11:52 +01:00
|
|
|
#include <cstddef>
|
2019-12-30 17:28:35 -06:00
|
|
|
#include <cstdlib>
|
2022-08-19 19:50:36 -07:00
|
|
|
#include <filesystem>
|
2010-11-10 04:12:31 +00:00
|
|
|
#include <iomanip>
|
2020-03-05 21:00:28 +00:00
|
|
|
#include <limits>
|
2019-12-30 17:28:35 -06:00
|
|
|
#include <locale>
|
2023-08-14 21:02:57 -04:00
|
|
|
#include <span>
|
2014-02-17 05:18:15 -05:00
|
|
|
#include <sstream>
|
|
|
|
#include <string>
|
2018-06-03 14:10:52 +02:00
|
|
|
#include <type_traits>
|
2014-02-17 05:18:15 -05:00
|
|
|
#include <vector>
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2015-09-26 17:13:07 -04:00
|
|
|
#include "Common/CommonTypes.h"
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2023-06-07 21:11:05 -04:00
|
|
|
namespace detail
|
|
|
|
{
|
|
|
|
template <typename T>
|
|
|
|
constexpr bool IsBooleanEnum()
|
|
|
|
{
|
|
|
|
if constexpr (std::is_enum_v<T>)
|
|
|
|
{
|
|
|
|
return std::is_same_v<std::underlying_type_t<T>, bool>;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} // namespace detail
|
|
|
|
|
2014-12-29 01:09:07 +01:00
|
|
|
std::string StringFromFormatV(const char* format, va_list args);
|
|
|
|
|
2014-02-07 01:10:04 +13: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-07-12 17:40:22 +00:00
|
|
|
// Cheap!
|
|
|
|
bool CharArrayFromFormatV(char* out, int outsize, const char* format, va_list args);
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2008-07-12 17:40:22 +00:00
|
|
|
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);
|
|
|
|
}
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2010-11-10 04:12:31 +00: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 04:12:31 +00:00
|
|
|
|
2022-07-19 15:13:26 -07:00
|
|
|
std::string_view StripWhitespace(std::string_view s);
|
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 04:12:31 +00:00
|
|
|
|
2019-12-30 17:28:35 -06:00
|
|
|
std::string ReplaceAll(std::string result, std::string_view src, std::string_view dest);
|
|
|
|
|
2021-05-11 16:30:29 +03:00
|
|
|
void ReplaceBreaksWithSpaces(std::string& str);
|
|
|
|
|
2022-07-18 21:45:27 -07:00
|
|
|
void TruncateToCString(std::string* s);
|
|
|
|
|
2016-01-21 21:27:56 +01:00
|
|
|
bool TryParse(const std::string& str, bool* output);
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2023-06-07 21:11:05 -04:00
|
|
|
template <typename T>
|
|
|
|
requires(std::is_integral_v<T> ||
|
|
|
|
(std::is_enum_v<T> && !detail::IsBooleanEnum<T>())) bool TryParse(const std::string& str,
|
|
|
|
T* output, int base = 0)
|
2010-11-10 04:12:31 +00:00
|
|
|
{
|
2019-12-30 17:28:35 -06: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-30 17:28:35 -06:00
|
|
|
if constexpr (std::is_unsigned_v<T>)
|
2020-05-23 18:40:10 -05:00
|
|
|
value = std::strtoull(str.c_str(), &end_ptr, base);
|
2019-12-30 17:28:35 -06:00
|
|
|
else
|
2020-05-23 18:40:10 -05:00
|
|
|
value = std::strtoll(str.c_str(), &end_ptr, base);
|
2019-12-30 17:28:35 -06: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;
|
|
|
|
}
|
|
|
|
|
2023-06-07 21:11:05 -04:00
|
|
|
template <typename T>
|
|
|
|
requires(detail::IsBooleanEnum<T>()) bool TryParse(const std::string& str, T* output)
|
|
|
|
{
|
|
|
|
bool value;
|
|
|
|
if (!TryParse(str, &value))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
*output = static_cast<T>(value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-12-30 17:28:35 -06:00
|
|
|
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-18 19:56:48 -05:00
|
|
|
if (iss >> tmp && iss.eof())
|
2010-11-10 04:12:31 +00:00
|
|
|
{
|
|
|
|
*output = tmp;
|
|
|
|
return true;
|
|
|
|
}
|
2018-06-15 08:11:18 -04:00
|
|
|
|
|
|
|
return false;
|
2010-11-10 04:12:31 +00:00
|
|
|
}
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2014-07-29 11:34:57 -05: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 11:34:57 -05: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);
|
|
|
|
|
2023-06-10 07:56:40 -05:00
|
|
|
namespace Common
|
|
|
|
{
|
|
|
|
template <typename T, typename std::enable_if_t<std::is_integral_v<T>>* = nullptr>
|
|
|
|
std::from_chars_result FromChars(std::string_view sv, T& value, int base = 10)
|
|
|
|
{
|
|
|
|
const char* const first = sv.data();
|
|
|
|
const char* const last = first + sv.size();
|
|
|
|
return std::from_chars(first, last, value, base);
|
|
|
|
}
|
|
|
|
template <typename T, typename std::enable_if_t<std::is_floating_point_v<T>>* = nullptr>
|
|
|
|
std::from_chars_result FromChars(std::string_view sv, T& value,
|
|
|
|
std::chars_format fmt = std::chars_format::general)
|
|
|
|
{
|
|
|
|
const char* const first = sv.data();
|
|
|
|
const char* const last = first + sv.size();
|
|
|
|
return std::from_chars(first, last, value, fmt);
|
|
|
|
}
|
2024-08-18 15:08:44 +02:00
|
|
|
} // namespace Common
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2019-07-08 13:35:53 +02:00
|
|
|
std::string TabsToSpaces(int tab_size, std::string str);
|
2009-06-21 08:39:21 +00:00
|
|
|
|
2017-06-11 16:33:10 +02:00
|
|
|
std::vector<std::string> SplitString(const std::string& str, char delim);
|
2008-12-08 04:46:09 +00:00
|
|
|
|
2010-11-10 04:12:31 +00:00
|
|
|
// "C:/Windows/winhelp.exe" to "C:/Windows/", "winhelp", ".exe"
|
2022-04-16 02:01:24 +02:00
|
|
|
// This requires forward slashes to be used for the path separators, even on Windows.
|
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 11:48:49 +00:00
|
|
|
|
2022-04-16 02:01:24 +02:00
|
|
|
// Converts the path separators of a path into forward slashes on Windows, which is assumed to be
|
|
|
|
// true for paths at various places in the codebase.
|
|
|
|
void UnifyPathSeparators(std::string& path);
|
|
|
|
std::string WithUnifiedPathSeparators(std::string path);
|
|
|
|
|
|
|
|
// Extracts just the filename (including extension) from a full path.
|
|
|
|
// This requires forward slashes to be used for the path separators, even on Windows.
|
2020-03-16 21:03:21 +01:00
|
|
|
std::string PathToFileName(std::string_view path);
|
|
|
|
|
2017-06-06 05:08:51 +01:00
|
|
|
void StringPopBackIf(std::string* s, char c);
|
2023-01-24 16:55:41 -05:00
|
|
|
size_t StringUTF8CodePointCount(std::string_view str);
|
2016-12-06 15:43:41 +00: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-02 19:46:55 -06:00
|
|
|
|
2013-02-27 18:00:42 -06:00
|
|
|
#ifdef _WIN32
|
|
|
|
|
2020-06-28 18:15:30 +02:00
|
|
|
std::wstring UTF8ToWString(std::string_view str);
|
2013-02-27 18:00:42 -06:00
|
|
|
|
2013-02-27 18:51:02 -06:00
|
|
|
#ifdef _UNICODE
|
2019-07-08 13:35:53 +02:00
|
|
|
inline std::string TStrToUTF8(std::wstring_view str)
|
2013-02-27 18:51:02 -06:00
|
|
|
{
|
2020-06-28 18:15:30 +02:00
|
|
|
return WStringToUTF8(str);
|
2013-02-27 18:51:02 -06:00
|
|
|
}
|
|
|
|
|
2019-07-08 13:35:53 +02:00
|
|
|
inline std::wstring UTF8ToTStr(std::string_view str)
|
2013-02-27 18:51:02 -06:00
|
|
|
{
|
2020-06-28 18:15:30 +02:00
|
|
|
return UTF8ToWString(str);
|
2013-02-27 18:51:02 -06:00
|
|
|
}
|
|
|
|
#else
|
2019-07-08 13:35:53 +02:00
|
|
|
inline std::string TStrToUTF8(std::string_view str)
|
2013-02-27 18:51:02 -06:00
|
|
|
{
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2019-07-08 13:35:53 +02:00
|
|
|
inline std::string UTF8ToTStr(std::string_view str)
|
2013-02-27 18:51:02 -06:00
|
|
|
{
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2013-02-27 18:00:42 -06:00
|
|
|
#endif
|
2018-05-08 13:55:07 +02:00
|
|
|
|
2019-06-21 18:07:59 +02:00
|
|
|
std::filesystem::path StringToPath(std::string_view path);
|
|
|
|
std::string PathToString(const std::filesystem::path& path);
|
|
|
|
|
2022-01-16 16:23:12 -08:00
|
|
|
namespace Common
|
|
|
|
{
|
2023-05-16 14:17:54 -04: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());
|
|
|
|
}
|
|
|
|
|
2023-05-16 14:11:02 -04:00
|
|
|
/// Returns whether a character is a letter, i.e. whether 'a' <= c <= 'z' || 'A' <= c <= 'Z'
|
|
|
|
/// is true. Use this instead of calling std::isalpha directly to ensure
|
|
|
|
/// the C locale is being used and to avoid possibly undefined behaviour.
|
|
|
|
inline bool IsAlpha(char c)
|
|
|
|
{
|
|
|
|
return std::isalpha(c, std::locale::classic());
|
|
|
|
}
|
|
|
|
|
2022-01-16 16:23:12 -08:00
|
|
|
inline char ToLower(char ch)
|
|
|
|
{
|
|
|
|
return std::tolower(ch, std::locale::classic());
|
|
|
|
}
|
2023-05-16 14:11:02 -04:00
|
|
|
|
2022-01-16 16:23:12 -08:00
|
|
|
inline char ToUpper(char ch)
|
|
|
|
{
|
|
|
|
return std::toupper(ch, std::locale::classic());
|
|
|
|
}
|
2023-05-16 14:11:02 -04:00
|
|
|
|
2023-05-16 14:26:33 -04: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
|
|
|
|
return WStringToUTF8(stream.str());
|
|
|
|
#else
|
|
|
|
return stream.str();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2023-05-16 14:23:21 -04:00
|
|
|
#ifdef _WIN32
|
|
|
|
std::vector<std::string> CommandLineToUtf8Argv(const wchar_t* command_line);
|
|
|
|
#endif
|
|
|
|
|
2023-05-16 14:21:19 -04:00
|
|
|
std::string GetEscapedHtml(std::string html);
|
|
|
|
|
2022-01-16 16:23:12 -08:00
|
|
|
void ToLower(std::string* str);
|
|
|
|
void ToUpper(std::string* str);
|
2022-05-31 02:06:42 +02:00
|
|
|
bool CaseInsensitiveEquals(std::string_view a, std::string_view b);
|
2024-09-21 22:32:24 -07:00
|
|
|
|
|
|
|
// 'std::less'-like comparison function object type for case-insensitive strings.
|
|
|
|
struct CaseInsensitiveLess
|
|
|
|
{
|
|
|
|
using is_transparent = void; // Allow heterogenous lookup.
|
|
|
|
bool operator()(std::string_view a, std::string_view b) const;
|
|
|
|
};
|
|
|
|
|
2023-08-14 21:02:57 -04:00
|
|
|
std::string BytesToHexString(std::span<const u8> bytes);
|
2022-01-16 16:23:12 -08:00
|
|
|
} // namespace Common
|