From c0730d3be02031246ea3f16a6ead03bfd59e952b Mon Sep 17 00:00:00 2001 From: Victor Zverovich Date: Sun, 19 Jan 2020 13:22:49 -0800 Subject: [PATCH] Add variable-width fill support (#1109) --- include/fmt/chrono.h | 2 +- include/fmt/format.h | 72 +++++++++++++++++++++++++++++++++----------- test/format-test.cc | 3 +- 3 files changed, 58 insertions(+), 19 deletions(-) diff --git a/include/fmt/chrono.h b/include/fmt/chrono.h index a77d86ad65366..dd003202a14f4 100644 --- a/include/fmt/chrono.h +++ b/include/fmt/chrono.h @@ -1024,7 +1024,7 @@ struct formatter, Char> { } void on_error(const char* msg) { FMT_THROW(format_error(msg)); } - void on_fill(Char fill) { f.specs.fill[0] = fill; } + void on_fill(basic_string_view fill) { f.specs.fill = fill; } void on_align(align_t align) { f.specs.align = align; } void on_width(int width) { f.specs.width = width; } void on_precision(int _precision) { f.precision = _precision; } diff --git a/include/fmt/format.h b/include/fmt/format.h index b6a26f3bddae5..a55b4a04d8db2 100644 --- a/include/fmt/format.h +++ b/include/fmt/format.h @@ -962,9 +962,25 @@ template struct null {}; // Workaround an array initialization issue in gcc 4.8. template struct fill_t { private: - Char data_[6]; + enum { max_size = 5 }; + Char data_[max_size + 1]; public: + FMT_CONSTEXPR void operator=(basic_string_view s) { + if (s.size() > max_size) throw format_error("invalid fill"); + auto size = s.size(); + for (size_t i = 0; i < size; ++i) data_[i] = s[i]; + data_[size] = Char(); + } + + size_t size() const { + size_t i = 1; + while (data_[i] && i <= max_size) ++i; + return i; + } + + const Char* data() const { return data_; } + FMT_CONSTEXPR Char& operator[](size_t index) { return data_[index]; } FMT_CONSTEXPR const Char& operator[](size_t index) const { return data_[index]; @@ -1352,6 +1368,14 @@ template struct nonfinite_writer { } }; +template +OutputIt fill(OutputIt it, size_t n, const fill_t& fill) { + auto fill_size = fill.size(); + if (fill_size == 1) return std::fill_n(it, n, fill[0]); + for (size_t i = 0; i < n; ++i) it = std::copy_n(fill.data(), fill_size, it); + return it; +} + // This template provides operations for formatting and writing data into a // character range. template class basic_writer { @@ -1614,20 +1638,20 @@ template class basic_writer { size_t size = f.size(); // The number of code units. size_t num_code_points = width != 0 ? f.width() : size; if (width <= num_code_points) return f(reserve(size)); - auto&& it = reserve(width + (size - num_code_points)); - char_type fill = specs.fill[0]; - std::size_t padding = width - num_code_points; + size_t padding = width - num_code_points; + size_t fill_size = specs.fill.size(); + auto&& it = reserve(size + padding * fill_size); if (specs.align == align::right) { - it = std::fill_n(it, padding, fill); + it = fill(it, padding, specs.fill); f(it); } else if (specs.align == align::center) { std::size_t left_padding = padding / 2; - it = std::fill_n(it, left_padding, fill); + it = fill(it, left_padding, specs.fill); f(it); - it = std::fill_n(it, padding - left_padding, fill); + it = fill(it, padding - left_padding, specs.fill); } else { f(it); - it = std::fill_n(it, padding, fill); + it = fill(it, padding, specs.fill); } } @@ -2008,7 +2032,9 @@ template class specs_setter { : specs_(other.specs_) {} FMT_CONSTEXPR void on_align(align_t align) { specs_.align = align; } - FMT_CONSTEXPR void on_fill(Char fill) { specs_.fill[0] = fill; } + FMT_CONSTEXPR void on_fill(basic_string_view fill) { + specs_.fill = fill; + } FMT_CONSTEXPR void on_plus() { specs_.sign = sign::plus; } FMT_CONSTEXPR void on_minus() { specs_.sign = sign::minus; } FMT_CONSTEXPR void on_space() { specs_.sign = sign::space; } @@ -2320,16 +2346,25 @@ template struct precision_adapter { SpecHandler& handler; }; +template +FMT_CONSTEXPR const Char* next_code_point(const Char* begin, const Char* end) { + if (sizeof(Char) != 1 || (*begin & 0x80) == 0) return begin + 1; + do { + ++begin; + } while (begin != end && (*begin & 0xc0) == 0x80); + return begin; +} + // Parses fill and alignment. template FMT_CONSTEXPR const Char* parse_align(const Char* begin, const Char* end, Handler&& handler) { FMT_ASSERT(begin != end, ""); auto align = align::none; - int i = 0; - if (begin + 1 != end) ++i; - do { - switch (static_cast(begin[i])) { + auto p = next_code_point(begin, end); + if (p == end) p = begin; + for (;;) { + switch (static_cast(*p)) { case '<': align = align::left; break; @@ -2346,18 +2381,21 @@ FMT_CONSTEXPR const Char* parse_align(const Char* begin, const Char* end, break; } if (align != align::none) { - if (i > 0) { + if (p != begin) { auto c = *begin; if (c == '{') return handler.on_error("invalid fill character '{'"), begin; - begin += 2; - handler.on_fill(c); + handler.on_fill(basic_string_view(begin, p - begin)); + begin = p + 1; } else ++begin; handler.on_align(align); break; + } else if (p == begin) { + break; } - } while (i-- > 0); + p = begin; + } return begin; } diff --git a/test/format-test.cc b/test/format-test.cc index 76f5710651f14..015a7438d5814 100644 --- a/test/format-test.cc +++ b/test/format-test.cc @@ -815,6 +815,7 @@ TEST(FormatterTest, Fill) { EXPECT_EQ("**0xface", format("{0:*>8}", reinterpret_cast(0xface))); EXPECT_EQ("foo=", format("{:}=", "foo")); EXPECT_EQ(std::string("\0\0\0*", 4), format(string_view("{:\0>4}", 6), '*')); + EXPECT_EQ("жж42", format("{0:ж>4}", 42)); } TEST(FormatterTest, PlusSign) { @@ -2173,7 +2174,7 @@ struct test_format_specs_handler { type(other.type) {} FMT_CONSTEXPR void on_align(fmt::align_t a) { align = a; } - FMT_CONSTEXPR void on_fill(char f) { fill = f; } + FMT_CONSTEXPR void on_fill(fmt::string_view f) { fill = f[0]; } FMT_CONSTEXPR void on_plus() { res = PLUS; } FMT_CONSTEXPR void on_minus() { res = MINUS; } FMT_CONSTEXPR void on_space() { res = SPACE; }