1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-10-20 19:46:35 +00:00
openmw/components/misc/strings/format.hpp
2022-09-22 21:35:26 +03:00

77 lines
2.5 KiB
C++

#ifndef COMPONENTS_MISC_STRINGS_FORMAT_H
#define COMPONENTS_MISC_STRINGS_FORMAT_H
#include <cerrno>
#include <cstdio>
#include <cstring>
#include <stdexcept>
#include <string>
#include <string_view>
#include <vector>
namespace Misc::StringUtils
{
namespace Details
{
// Allow to convert complex arguments to C-style strings for format() function
template <typename T>
T argument(T value) noexcept
{
static_assert(!std::is_same_v<T, std::string_view>, "std::string_view is not supported");
return value;
}
template <typename T>
T const* argument(std::basic_string<T> const& value) noexcept
{
return value.c_str();
}
template <class T>
T nullTerminated(T value) noexcept
{
return value;
}
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)
throw std::runtime_error(
std::string("Failed to compute resulting string size: ") + std::strerror(errno));
// 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)
throw std::runtime_error(std::string("Failed to format string: ") + std::strerror(errno));
ret.erase(static_cast<std::size_t>(size));
return ret;
}
}
template <typename... Args>
std::string format(const char* fmt, Args const&... args)
{
return Details::format(fmt, Details::nullTerminated(args)...);
}
template <typename... Args>
std::string format(const std::string& fmt, Args const&... args)
{
return Details::format(fmt.c_str(), Details::nullTerminated(args)...);
}
}
#endif