mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-18 12:01:15 +01:00
574477866f
The control expression editor allows line breaks, but the serialization was losing anything after the first line break (/r /n). Instead of opting to encode them and decode them on serialization (which I tried but was not safe, as it would lose /n written in the string by users), I opted to replace them with a space.
245 lines
6.6 KiB
C++
245 lines
6.6 KiB
C++
// Copyright 2008 Dolphin Emulator Project
|
|
// Licensed under GPLv2+
|
|
// Refer to the license.txt file included.
|
|
|
|
#pragma once
|
|
|
|
#include <cstdarg>
|
|
#include <cstddef>
|
|
#include <cstdlib>
|
|
#include <iomanip>
|
|
#include <limits>
|
|
#include <locale>
|
|
#include <sstream>
|
|
#include <string>
|
|
#include <type_traits>
|
|
#include <vector>
|
|
|
|
#include "Common/CommonTypes.h"
|
|
|
|
#ifdef _MSC_VER
|
|
#include <filesystem>
|
|
#define HAS_STD_FILESYSTEM
|
|
#endif
|
|
|
|
std::string StringFromFormatV(const char* format, va_list args);
|
|
|
|
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
|
|
;
|
|
|
|
// 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);
|
|
}
|
|
|
|
// Good
|
|
std::string ArrayToString(const u8* data, u32 size, int line_len = 20, bool spaces = true);
|
|
|
|
std::string_view StripSpaces(std::string_view s);
|
|
std::string_view StripQuotes(std::string_view s);
|
|
|
|
std::string ReplaceAll(std::string result, std::string_view src, std::string_view dest);
|
|
|
|
void ReplaceBreaksWithSpaces(std::string& str);
|
|
|
|
bool TryParse(const std::string& str, bool* output);
|
|
|
|
template <typename T, std::enable_if_t<std::is_integral_v<T> || std::is_enum_v<T>>* = nullptr>
|
|
bool TryParse(const std::string& str, T* output, int base = 0)
|
|
{
|
|
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;
|
|
|
|
if constexpr (std::is_unsigned_v<T>)
|
|
value = std::strtoull(str.c_str(), &end_ptr, base);
|
|
else
|
|
value = std::strtoll(str.c_str(), &end_ptr, base);
|
|
|
|
// 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.
|
|
if (iss >> tmp && iss.eof())
|
|
{
|
|
*output = tmp;
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
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;
|
|
|
|
while (std::getline(buffer, variable, delimiter))
|
|
{
|
|
N tmp = 0;
|
|
if (!TryParse(variable, &tmp))
|
|
return false;
|
|
output->push_back(tmp);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
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));
|
|
}
|
|
|
|
// Generates an hexdump-like representation of a binary data blob.
|
|
std::string HexDump(const u8* data, size_t size);
|
|
|
|
// TODO: kill this
|
|
bool AsciiToHex(const std::string& _szValue, u32& result);
|
|
|
|
std::string TabsToSpaces(int tab_size, std::string str);
|
|
|
|
std::vector<std::string> SplitString(const std::string& str, char delim);
|
|
std::string JoinStrings(const std::vector<std::string>& strings, const std::string& delimiter);
|
|
|
|
// "C:/Windows/winhelp.exe" to "C:/Windows/", "winhelp", ".exe"
|
|
bool SplitPath(std::string_view full_path, std::string* path, std::string* filename,
|
|
std::string* extension);
|
|
|
|
std::string PathToFileName(std::string_view path);
|
|
|
|
void BuildCompleteFilename(std::string& complete_filename, std::string_view path,
|
|
std::string_view filename);
|
|
|
|
bool StringBeginsWith(std::string_view str, std::string_view begin);
|
|
bool StringEndsWith(std::string_view str, std::string_view end);
|
|
void StringPopBackIf(std::string* s, char c);
|
|
size_t StringUTF8CodePointCount(const std::string& str);
|
|
|
|
std::string CP1252ToUTF8(std::string_view str);
|
|
std::string SHIFTJISToUTF8(std::string_view str);
|
|
std::string UTF8ToSHIFTJIS(std::string_view str);
|
|
std::string WStringToUTF8(std::wstring_view str);
|
|
std::string UTF16BEToUTF8(const char16_t* str, size_t max_size); // Stops at \0
|
|
std::string UTF16ToUTF8(std::u16string_view str);
|
|
std::u16string UTF8ToUTF16(std::string_view str);
|
|
|
|
#ifdef _WIN32
|
|
|
|
std::wstring UTF8ToWString(std::string_view str);
|
|
|
|
#ifdef _UNICODE
|
|
inline std::string TStrToUTF8(std::wstring_view str)
|
|
{
|
|
return WStringToUTF8(str);
|
|
}
|
|
|
|
inline std::wstring UTF8ToTStr(std::string_view str)
|
|
{
|
|
return UTF8ToWString(str);
|
|
}
|
|
#else
|
|
inline std::string TStrToUTF8(std::string_view str)
|
|
{
|
|
return str;
|
|
}
|
|
|
|
inline std::string UTF8ToTStr(std::string_view str)
|
|
{
|
|
return str;
|
|
}
|
|
#endif
|
|
|
|
#endif
|
|
|
|
#ifdef HAS_STD_FILESYSTEM
|
|
std::filesystem::path StringToPath(std::string_view path);
|
|
std::string PathToString(const std::filesystem::path& path);
|
|
#endif
|
|
|
|
// 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
|
|
}
|
|
|
|
/// 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());
|
|
}
|
|
|
|
#ifdef _WIN32
|
|
std::vector<std::string> CommandLineToUtf8Argv(const wchar_t* command_line);
|
|
#endif
|