2022-08-03 00:00:54 +02:00
|
|
|
#ifndef COMPONENTS_MISC_STRINGS_FORMAT_H
|
|
|
|
#define COMPONENTS_MISC_STRINGS_FORMAT_H
|
|
|
|
|
2023-05-30 17:35:26 +02:00
|
|
|
#include <MyGUI_UString.h>
|
2022-08-03 00:00:54 +02:00
|
|
|
#include <cerrno>
|
|
|
|
#include <cstdio>
|
|
|
|
#include <cstring>
|
|
|
|
#include <stdexcept>
|
|
|
|
#include <string>
|
|
|
|
#include <string_view>
|
2023-01-29 21:56:59 +01:00
|
|
|
#include <system_error>
|
2022-08-24 23:10:05 +02:00
|
|
|
#include <vector>
|
2022-08-03 00:00:54 +02:00
|
|
|
|
|
|
|
namespace Misc::StringUtils
|
|
|
|
{
|
2022-08-26 19:43:57 +00:00
|
|
|
namespace Details
|
2022-08-03 00:00:54 +02:00
|
|
|
{
|
|
|
|
// Allow to convert complex arguments to C-style strings for format() function
|
|
|
|
template <typename T>
|
|
|
|
T argument(T value) noexcept
|
|
|
|
{
|
2022-08-26 19:43:57 +00:00
|
|
|
static_assert(!std::is_same_v<T, std::string_view>, "std::string_view is not supported");
|
2023-05-30 17:35:26 +02:00
|
|
|
static_assert(!std::is_same_v<T, MyGUI::UString>, "MyGUI::UString is not supported");
|
2022-08-03 00:00:54 +02:00
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2022-08-16 21:15:03 +02:00
|
|
|
template <typename T>
|
2022-08-26 19:43:57 +00:00
|
|
|
T const* argument(std::basic_string<T> const& value) noexcept
|
2022-08-16 21:15:03 +02:00
|
|
|
{
|
2022-08-26 19:43:57 +00:00
|
|
|
return value.c_str();
|
2022-08-16 21:15:03 +02:00
|
|
|
}
|
|
|
|
|
2022-08-26 19:43:57 +00:00
|
|
|
template <class T>
|
|
|
|
T nullTerminated(T value) noexcept
|
2022-08-03 00:00:54 +02:00
|
|
|
{
|
2022-08-26 19:43:57 +00:00
|
|
|
return value;
|
2022-08-03 00:00:54 +02:00
|
|
|
}
|
|
|
|
|
2022-08-26 19:43:57 +00:00
|
|
|
template <class T>
|
|
|
|
std::basic_string<T> nullTerminated(const std::basic_string_view<T>& value) noexcept
|
|
|
|
{
|
|
|
|
// Ensure string_view arguments are null-terminated by creating a string
|
|
|
|
// TODO: Use a format function that doesn't require this workaround
|
|
|
|
return std::string{ value };
|
|
|
|
}
|
|
|
|
|
|
|
|
// Requires some C++11 features:
|
|
|
|
// 1. std::string needs to be contiguous
|
|
|
|
// 2. std::snprintf with zero size (second argument) returns an output string size
|
|
|
|
// 3. variadic templates support
|
|
|
|
template <typename... Args>
|
|
|
|
std::string format(const char* fmt, Args const&... args)
|
|
|
|
{
|
|
|
|
const int size = std::snprintf(nullptr, 0, fmt, argument(args)...);
|
|
|
|
if (size < 0)
|
2023-01-29 21:56:59 +01:00
|
|
|
throw std::system_error(errno, std::generic_category(), "Failed to compute resulting string size");
|
2022-08-26 19:43:57 +00:00
|
|
|
// Note: sprintf also writes a trailing null character. We should remove it.
|
|
|
|
std::string ret(static_cast<std::size_t>(size) + 1, '\0');
|
|
|
|
if (std::sprintf(ret.data(), fmt, argument(args)...) < 0)
|
2023-01-29 21:56:59 +01:00
|
|
|
throw std::system_error(errno, std::generic_category(), "Failed to format string");
|
2022-08-26 19:43:57 +00:00
|
|
|
ret.erase(static_cast<std::size_t>(size));
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-03 00:00:54 +02:00
|
|
|
template <typename... Args>
|
|
|
|
std::string format(const char* fmt, Args const&... args)
|
|
|
|
{
|
2022-08-26 19:43:57 +00:00
|
|
|
return Details::format(fmt, Details::nullTerminated(args)...);
|
2022-08-03 00:00:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
template <typename... Args>
|
2022-08-24 23:10:05 +02:00
|
|
|
std::string format(const std::string& fmt, Args const&... args)
|
2022-08-03 00:00:54 +02:00
|
|
|
{
|
2022-08-26 19:43:57 +00:00
|
|
|
return Details::format(fmt.c_str(), Details::nullTerminated(args)...);
|
2022-08-03 00:00:54 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|