mirror of
https://github.com/fmtlib/fmt.git
synced 2025-07-30 02:37:36 +02:00
Support character range formatting (#3863)
This commit is contained in:
@ -13,7 +13,7 @@
|
|||||||
#include <tuple>
|
#include <tuple>
|
||||||
#include <type_traits>
|
#include <type_traits>
|
||||||
|
|
||||||
#include "base.h"
|
#include "format.h"
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
FMT_BEGIN_NAMESPACE
|
||||||
|
|
||||||
@ -388,6 +388,8 @@ struct range_formatter<
|
|||||||
detail::string_literal<Char, '['>{};
|
detail::string_literal<Char, '['>{};
|
||||||
basic_string_view<Char> closing_bracket_ =
|
basic_string_view<Char> closing_bracket_ =
|
||||||
detail::string_literal<Char, ']'>{};
|
detail::string_literal<Char, ']'>{};
|
||||||
|
bool is_string_format = false;
|
||||||
|
bool is_debug = false;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
FMT_CONSTEXPR range_formatter() {}
|
FMT_CONSTEXPR range_formatter() {}
|
||||||
@ -410,31 +412,79 @@ struct range_formatter<
|
|||||||
FMT_CONSTEXPR auto parse(ParseContext& ctx) -> decltype(ctx.begin()) {
|
FMT_CONSTEXPR auto parse(ParseContext& ctx) -> decltype(ctx.begin()) {
|
||||||
auto it = ctx.begin();
|
auto it = ctx.begin();
|
||||||
auto end = ctx.end();
|
auto end = ctx.end();
|
||||||
|
detail::maybe_set_debug_format(underlying_, true);
|
||||||
|
if (it == end) {
|
||||||
|
return underlying_.parse(ctx);
|
||||||
|
}
|
||||||
|
|
||||||
if (it != end && *it == 'n') {
|
switch (detail::to_ascii(*it)) {
|
||||||
|
case 'n':
|
||||||
set_brackets({}, {});
|
set_brackets({}, {});
|
||||||
++it;
|
++it;
|
||||||
|
break;
|
||||||
|
case '?':
|
||||||
|
is_debug = true;
|
||||||
|
set_brackets({}, {});
|
||||||
|
++it;
|
||||||
|
if (it == end || *it != 's') {
|
||||||
|
report_error("invalid format specifier");
|
||||||
|
}
|
||||||
|
FMT_FALLTHROUGH;
|
||||||
|
case 's':
|
||||||
|
if (!std::is_same<T, Char>::value) {
|
||||||
|
report_error("invalid format specifier");
|
||||||
|
}
|
||||||
|
if (!is_debug) {
|
||||||
|
set_brackets(detail::string_literal<Char, '"'>{},
|
||||||
|
detail::string_literal<Char, '"'>{});
|
||||||
|
set_separator({});
|
||||||
|
detail::maybe_set_debug_format(underlying_, false);
|
||||||
|
}
|
||||||
|
is_string_format = true;
|
||||||
|
++it;
|
||||||
|
return it;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (it != end && *it != '}') {
|
if (it != end && *it != '}') {
|
||||||
if (*it != ':') report_error("invalid format specifier");
|
if (*it != ':') report_error("invalid format specifier");
|
||||||
|
detail::maybe_set_debug_format(underlying_, false);
|
||||||
++it;
|
++it;
|
||||||
} else {
|
|
||||||
detail::maybe_set_debug_format(underlying_, true);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx.advance_to(it);
|
ctx.advance_to(it);
|
||||||
return underlying_.parse(ctx);
|
return underlying_.parse(ctx);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template <typename Output, typename Iter, typename IterEnd, typename U = T,
|
||||||
|
FMT_ENABLE_IF(std::is_same<U, Char>::value)>
|
||||||
|
auto write_debug_string(Output& out, Iter& it, IterEnd& end) const -> Output {
|
||||||
|
auto buf = basic_memory_buffer<Char>();
|
||||||
|
for (; it != end; ++it) {
|
||||||
|
buf.push_back(*it);
|
||||||
|
}
|
||||||
|
format_specs spec_str;
|
||||||
|
spec_str.type = presentation_type::debug;
|
||||||
|
return detail::write<Char>(
|
||||||
|
out, basic_string_view<Char>(buf.data(), buf.size()), spec_str);
|
||||||
|
}
|
||||||
|
template <typename Output, typename Iter, typename IterEnd, typename U = T,
|
||||||
|
FMT_ENABLE_IF(!std::is_same<U, Char>::value)>
|
||||||
|
auto write_debug_string(Output& out, Iter&, IterEnd&) const -> Output {
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
template <typename R, typename FormatContext>
|
template <typename R, typename FormatContext>
|
||||||
auto format(R&& range, FormatContext& ctx) const -> decltype(ctx.out()) {
|
auto format(R&& range, FormatContext& ctx) const -> decltype(ctx.out()) {
|
||||||
detail::range_mapper<buffered_context<Char>> mapper;
|
detail::range_mapper<buffered_context<Char>> mapper;
|
||||||
auto out = ctx.out();
|
auto out = ctx.out();
|
||||||
out = detail::copy<Char>(opening_bracket_, out);
|
|
||||||
int i = 0;
|
|
||||||
auto it = detail::range_begin(range);
|
auto it = detail::range_begin(range);
|
||||||
auto end = detail::range_end(range);
|
auto end = detail::range_end(range);
|
||||||
|
if (is_debug) {
|
||||||
|
return write_debug_string(out, it, end);
|
||||||
|
}
|
||||||
|
|
||||||
|
out = detail::copy<Char>(opening_bracket_, out);
|
||||||
|
int i = 0;
|
||||||
for (; it != end; ++it) {
|
for (; it != end; ++it) {
|
||||||
if (i > 0) out = detail::copy<Char>(separator_, out);
|
if (i > 0) out = detail::copy<Char>(separator_, out);
|
||||||
ctx.advance_to(out);
|
ctx.advance_to(out);
|
||||||
|
@ -58,8 +58,13 @@ TEST(ranges_test, format_vector) {
|
|||||||
EXPECT_EQ(fmt::format("{:n:#x}", v), "0x1, 0x2, 0x3, 0x5, 0x7, 0xb");
|
EXPECT_EQ(fmt::format("{:n:#x}", v), "0x1, 0x2, 0x3, 0x5, 0x7, 0xb");
|
||||||
|
|
||||||
auto vc = std::vector<char>{'a', 'b', 'c'};
|
auto vc = std::vector<char>{'a', 'b', 'c'};
|
||||||
|
auto vec = std::vector<char>{'a', '\n', '\t'};
|
||||||
auto vvc = std::vector<std::vector<char>>{vc, vc};
|
auto vvc = std::vector<std::vector<char>>{vc, vc};
|
||||||
EXPECT_EQ(fmt::format("{}", vc), "['a', 'b', 'c']");
|
EXPECT_EQ(fmt::format("{}", vc), "['a', 'b', 'c']");
|
||||||
|
EXPECT_EQ(fmt::format("{:s}", vc), "\"abc\"");
|
||||||
|
EXPECT_EQ(fmt::format("{:?s}", vec), "\"a\\n\\t\"");
|
||||||
|
EXPECT_EQ(fmt::format("{:s}", vec), "\"a\n\t\"");
|
||||||
|
EXPECT_EQ(fmt::format("{::s}", vvc), "[\"abc\", \"abc\"]");
|
||||||
EXPECT_EQ(fmt::format("{}", vvc), "[['a', 'b', 'c'], ['a', 'b', 'c']]");
|
EXPECT_EQ(fmt::format("{}", vvc), "[['a', 'b', 'c'], ['a', 'b', 'c']]");
|
||||||
EXPECT_EQ(fmt::format("{:n}", vvc), "['a', 'b', 'c'], ['a', 'b', 'c']");
|
EXPECT_EQ(fmt::format("{:n}", vvc), "['a', 'b', 'c'], ['a', 'b', 'c']");
|
||||||
EXPECT_EQ(fmt::format("{:n:n}", vvc), "'a', 'b', 'c', 'a', 'b', 'c'");
|
EXPECT_EQ(fmt::format("{:n:n}", vvc), "'a', 'b', 'c', 'a', 'b', 'c'");
|
||||||
|
Reference in New Issue
Block a user