Commit b0cde860 authored by Daniela Engert's avatar Daniela Engert Committed by Victor Zverovich

Implement 'snprintf(OutputIt it, size_t n, const S &format, const Args & ... args)' (#917)

Mostly equivalent to 'sprintf(const S &format, const Args & ... args)' but generates at most 'n' characters through output iterator 'it'. The output type is the same as with 'format_to_n'.
Signed-off-by: default avatarDaniela Engert <dani@ngrt.de>
parent e05dfb08
...@@ -192,6 +192,13 @@ void printf(basic_buffer<Char>& buf, basic_string_view<Char> format, ...@@ -192,6 +192,13 @@ void printf(basic_buffer<Char>& buf, basic_string_view<Char> format,
basic_format_args<Context> args) { basic_format_args<Context> args) {
Context(std::back_inserter(buf), format, args).format(); Context(std::back_inserter(buf), format, args).format();
} }
template <typename OutputIt, typename Char, typename Context>
internal::truncating_iterator<OutputIt> printf(
internal::truncating_iterator<OutputIt> it, basic_string_view<Char> format,
basic_format_args<Context> args) {
return Context(it, format, args).format();
}
} // namespace internal } // namespace internal
using internal::printf; // For printing into memory_buffer. using internal::printf; // For printing into memory_buffer.
...@@ -217,7 +224,8 @@ class printf_arg_formatter ...@@ -217,7 +224,8 @@ class printf_arg_formatter
typedef typename Range::value_type char_type; typedef typename Range::value_type char_type;
typedef decltype(internal::declval<Range>().begin()) iterator; typedef decltype(internal::declval<Range>().begin()) iterator;
typedef internal::arg_formatter_base<Range> base; typedef internal::arg_formatter_base<Range> base;
typedef basic_printf_context<iterator, char_type> context_type; typedef basic_printf_context<iterator, char_type, printf_arg_formatter>
context_type;
context_type& context_; context_type& context_;
...@@ -241,11 +249,8 @@ class printf_arg_formatter ...@@ -241,11 +249,8 @@ class printf_arg_formatter
specifier information for standard argument types. specifier information for standard argument types.
\endrst \endrst
*/ */
printf_arg_formatter(internal::basic_buffer<char_type>& buffer, printf_arg_formatter(iterator iter, format_specs& spec, context_type& ctx)
format_specs& spec, context_type& ctx) : base(Range(iter), &spec, ctx.locale()), context_(ctx) {}
: base(back_insert_range<internal::basic_buffer<char_type>>(buffer),
&spec, ctx.locale()),
context_(ctx) {}
template <typename T> template <typename T>
typename std::enable_if<std::is_integral<T>::value, iterator>::type typename std::enable_if<std::is_integral<T>::value, iterator>::type
...@@ -378,7 +383,7 @@ class basic_printf_context : ...@@ -378,7 +383,7 @@ class basic_printf_context :
using base::parse_context; using base::parse_context;
/** Formats stored arguments and writes the output to the range. */ /** Formats stored arguments and writes the output to the range. */
void format(); OutputIt format();
}; };
template <typename OutputIt, typename Char, typename AF> template <typename OutputIt, typename Char, typename AF>
...@@ -455,21 +460,21 @@ unsigned basic_printf_context<OutputIt, Char, AF>::parse_header( ...@@ -455,21 +460,21 @@ unsigned basic_printf_context<OutputIt, Char, AF>::parse_header(
} }
template <typename OutputIt, typename Char, typename AF> template <typename OutputIt, typename Char, typename AF>
void basic_printf_context<OutputIt, Char, AF>::format() { OutputIt basic_printf_context<OutputIt, Char, AF>::format() {
auto& buffer = internal::get_container(this->out()); auto out = this->out();
const auto range = this->parse_context(); const auto range = this->parse_context();
const Char* end = range.end(); const Char* const end = range.end();
const Char* start = range.begin(); const Char* start = range.begin();
auto it = start; auto it = start;
while (it != end) { while (it != end) {
char_type c = *it++; char_type c = *it++;
if (c != '%') continue; if (c != '%') continue;
if (it != end && *it == c) { if (it != end && *it == c) {
buffer.append(start, it); out = std::copy(start, it, out);
start = ++it; start = ++it;
continue; continue;
} }
buffer.append(start, it - 1); out = std::copy(start, it - 1, out);
format_specs spec; format_specs spec;
spec.align_ = ALIGN_RIGHT; spec.align_ = ALIGN_RIGHT;
...@@ -566,9 +571,9 @@ void basic_printf_context<OutputIt, Char, AF>::format() { ...@@ -566,9 +571,9 @@ void basic_printf_context<OutputIt, Char, AF>::format() {
start = it; start = it;
// Format argument. // Format argument.
visit_format_arg(AF(buffer, spec, *this), arg); visit_format_arg(AF(out, spec, *this), arg);
} }
buffer.append(start, it); return std::copy(start, it, out);
} }
template <typename Buffer> struct basic_printf_context_t { template <typename Buffer> struct basic_printf_context_t {
...@@ -583,6 +588,14 @@ typedef basic_printf_context_t<internal::wbuffer>::type wprintf_context; ...@@ -583,6 +588,14 @@ typedef basic_printf_context_t<internal::wbuffer>::type wprintf_context;
typedef basic_format_args<printf_context> printf_args; typedef basic_format_args<printf_context> printf_args;
typedef basic_format_args<wprintf_context> wprintf_args; typedef basic_format_args<wprintf_context> wprintf_args;
template <typename OutputIt, typename Char = typename OutputIt::value_type>
struct basic_printf_n_context_t {
typedef fmt::internal::truncating_iterator<OutputIt> OutputIter;
typedef output_range<OutputIter, Char> Range;
typedef basic_printf_context<OutputIter, Char, printf_arg_formatter<Range>>
type;
};
/** /**
\rst \rst
Constructs an `~fmt::format_arg_store` object that contains references to Constructs an `~fmt::format_arg_store` object that contains references to
...@@ -618,6 +631,18 @@ inline std::basic_string<Char> vsprintf( ...@@ -618,6 +631,18 @@ inline std::basic_string<Char> vsprintf(
return to_string(buffer); return to_string(buffer);
} }
template <typename OutputIt, typename S, typename Char = FMT_CHAR(S)>
inline typename std::enable_if<internal::is_output_iterator<OutputIt>::value,
format_to_n_result<OutputIt>>::type
vsnprintf(
OutputIt out, std::size_t n, const S& format,
basic_format_args<typename basic_printf_n_context_t<OutputIt, Char>::type>
args) {
typedef internal::truncating_iterator<OutputIt> It;
auto it = printf(It(out, n), to_string_view(format), args);
return {it.base(), it.count()};
}
/** /**
\rst \rst
Formats arguments and returns the result as a string. Formats arguments and returns the result as a string.
...@@ -638,6 +663,34 @@ inline FMT_ENABLE_IF_T(internal::is_string<S>::value, ...@@ -638,6 +663,34 @@ inline FMT_ENABLE_IF_T(internal::is_string<S>::value,
return vsprintf(to_string_view(format), basic_format_args<context>(as)); return vsprintf(to_string_view(format), basic_format_args<context>(as));
} }
/**
\rst
Formats arguments for up to ``n`` characters stored through output iterator
``out``. The function returns the updated iterator and the untruncated amount
of characters.
**Example**::
std::vector<char> out;
typedef fmt::format_to_n_result<
std::back_insert_iterator<std::vector<char>>> res;
res Res = fmt::snprintf(std::back_inserter(out), 5, "The answer is %d", 42);
\endrst
*/
template <typename OutputIt, typename S, typename... Args>
inline FMT_ENABLE_IF_T(internal::is_string<S>::value&&
internal::is_output_iterator<OutputIt>::value,
format_to_n_result<OutputIt>)
snprintf(OutputIt out, std::size_t n, const S& format,
const Args&... args) {
internal::check_format_string<Args...>(format);
typedef FMT_CHAR(S) Char;
typedef typename basic_printf_n_context_t<OutputIt, Char>::type context;
format_arg_store<context, Args...> as{args...};
return vsnprintf(out, n, to_string_view(format),
basic_format_args<context>(as));
}
template <typename S, typename Char = FMT_CHAR(S)> template <typename S, typename Char = FMT_CHAR(S)>
inline int vfprintf( inline int vfprintf(
std::FILE* f, const S& format, std::FILE* f, const S& format,
......
...@@ -555,3 +555,23 @@ TEST(PrintfTest, VSPrintfMakeWArgsExample) { ...@@ -555,3 +555,23 @@ TEST(PrintfTest, VSPrintfMakeWArgsExample) {
fmt::make_wprintf_args(42, L"something"))); fmt::make_wprintf_args(42, L"something")));
#endif #endif
} }
TEST(PrintfTest, snprintf) {
char buffer[4] = "xxx";
auto result = fmt::snprintf(buffer, 0, "test");
EXPECT_EQ("xxx", fmt::to_string_view(buffer));
EXPECT_EQ(4u, result.size);
EXPECT_EQ(buffer, result.out);
result = fmt::snprintf(buffer, 2, "test");
EXPECT_EQ("tex", fmt::to_string_view(buffer));
EXPECT_EQ(4u, result.size);
EXPECT_EQ(buffer + 2, result.out);
result = fmt::snprintf(buffer, 3, "test");
EXPECT_EQ("tes", fmt::to_string_view(buffer));
EXPECT_EQ(4u, result.size);
EXPECT_EQ(buffer + 3, result.out);
result = fmt::snprintf(buffer, 4, "test");
EXPECT_EQ("test", std::string(buffer, 4));
EXPECT_EQ(4u, result.size);
EXPECT_EQ(buffer + 4, result.out);
}
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment