// Formatting library for C++ - locale tests // // Copyright (c) 2012 - present, Victor Zverovich // All rights reserved. // // For the license information refer to format.h. #include "fmt/locale.h" #include #include "gmock/gmock.h" using fmt::detail::max_value; #ifndef FMT_STATIC_THOUSANDS_SEPARATOR template struct numpunct : std::numpunct { protected: Char do_decimal_point() const override { return '?'; } std::string do_grouping() const override { return "\03"; } Char do_thousands_sep() const override { return '~'; } }; template struct no_grouping : std::numpunct { protected: Char do_decimal_point() const override { return '.'; } std::string do_grouping() const override { return ""; } Char do_thousands_sep() const override { return ','; } }; template struct special_grouping : std::numpunct { protected: Char do_decimal_point() const override { return '.'; } std::string do_grouping() const override { return "\03\02"; } Char do_thousands_sep() const override { return ','; } }; template struct small_grouping : std::numpunct { protected: Char do_decimal_point() const override { return '.'; } std::string do_grouping() const override { return "\01"; } Char do_thousands_sep() const override { return ','; } }; TEST(locale_test, double_decimal_point) { auto loc = std::locale(std::locale(), new numpunct()); EXPECT_EQ("1?23", fmt::format(loc, "{:L}", 1.23)); EXPECT_EQ("1?230000", fmt::format(loc, "{:Lf}", 1.23)); } TEST(locale_test, format) { auto loc = std::locale(std::locale(), new numpunct()); EXPECT_EQ("1234567", fmt::format(std::locale(), "{:L}", 1234567)); EXPECT_EQ("1~234~567", fmt::format(loc, "{:L}", 1234567)); EXPECT_EQ("-1~234~567", fmt::format(loc, "{:L}", -1234567)); EXPECT_EQ("-256", fmt::format(loc, "{:L}", -256)); fmt::format_arg_store as{1234567}; EXPECT_EQ("1~234~567", fmt::vformat(loc, "{:L}", fmt::format_args(as))); auto s = std::string(); fmt::format_to(std::back_inserter(s), loc, "{:L}", 1234567); EXPECT_EQ("1~234~567", s); auto no_grouping_loc = std::locale(std::locale(), new no_grouping()); EXPECT_EQ("1234567", fmt::format(no_grouping_loc, "{:L}", 1234567)); auto special_grouping_loc = std::locale(std::locale(), new special_grouping()); EXPECT_EQ("1,23,45,678", fmt::format(special_grouping_loc, "{:L}", 12345678)); EXPECT_EQ("12,345", fmt::format(special_grouping_loc, "{:L}", 12345)); auto small_grouping_loc = std::locale(std::locale(), new small_grouping()); EXPECT_EQ("4,2,9,4,9,6,7,2,9,5", fmt::format(small_grouping_loc, "{:L}", max_value())); } TEST(locale_test, format_detault_align) { auto loc = std::locale({}, new special_grouping()); EXPECT_EQ(" 12,345", fmt::format(loc, "{:8L}", 12345)); } TEST(locale_test, format_plus) { auto loc = std::locale({}, new special_grouping()); EXPECT_EQ("+100", fmt::format(loc, "{:+L}", 100)); } TEST(locale_test, wformat) { auto loc = std::locale(std::locale(), new numpunct()); EXPECT_EQ(L"1234567", fmt::format(std::locale(), L"{:L}", 1234567)); EXPECT_EQ(L"1~234~567", fmt::format(loc, L"{:L}", 1234567)); using wcontext = fmt::buffer_context; fmt::format_arg_store as{1234567}; EXPECT_EQ(L"1~234~567", fmt::vformat(loc, L"{:L}", fmt::basic_format_args(as))); EXPECT_EQ(L"1234567", fmt::format(std::locale("C"), L"{:L}", 1234567)); auto no_grouping_loc = std::locale(std::locale(), new no_grouping()); EXPECT_EQ(L"1234567", fmt::format(no_grouping_loc, L"{:L}", 1234567)); auto special_grouping_loc = std::locale(std::locale(), new special_grouping()); EXPECT_EQ(L"1,23,45,678", fmt::format(special_grouping_loc, L"{:L}", 12345678)); auto small_grouping_loc = std::locale(std::locale(), new small_grouping()); EXPECT_EQ(L"4,2,9,4,9,6,7,2,9,5", fmt::format(small_grouping_loc, L"{:L}", max_value())); } TEST(locale_test, double_formatter) { auto loc = std::locale(std::locale(), new special_grouping()); auto f = fmt::formatter(); auto parse_ctx = fmt::format_parse_context("L"); f.parse(parse_ctx); char buf[10] = {}; fmt::basic_format_context format_ctx( buf, {}, fmt::detail::locale_ref(loc)); *f.format(12345, format_ctx) = 0; EXPECT_STREQ("12,345", buf); } FMT_BEGIN_NAMESPACE template struct formatter, charT> { private: detail::dynamic_format_specs specs_; public: FMT_CONSTEXPR typename basic_format_parse_context::iterator parse( basic_format_parse_context& ctx) { using handler_type = detail::dynamic_specs_handler>; detail::specs_checker handler(handler_type(specs_, ctx), detail::type::string_type); auto it = parse_format_specs(ctx.begin(), ctx.end(), handler); detail::parse_float_type_spec(specs_, ctx.error_handler()); return it; } template typename FormatContext::iterator format(const std::complex& c, FormatContext& ctx) { detail::handle_dynamic_spec( specs_.precision, specs_.precision_ref, ctx); auto specs = std::string(); if (specs_.precision > 0) specs = fmt::format(".{}", specs_.precision); if (specs_.type) specs += specs_.type; auto real = fmt::format(ctx.locale().template get(), "{:" + specs + "}", c.real()); auto imag = fmt::format(ctx.locale().template get(), "{:" + specs + "}", c.imag()); auto fill_align_width = std::string(); if (specs_.width > 0) fill_align_width = fmt::format(">{}", specs_.width); return format_to(ctx.out(), runtime("{:" + fill_align_width + "}"), c.real() != 0 ? fmt::format("({}+{}i)", real, imag) : fmt::format("{}i", imag)); } }; FMT_END_NAMESPACE TEST(locale_test, complex) { std::string s = fmt::format("{}", std::complex(1, 2)); EXPECT_EQ(s, "(1+2i)"); EXPECT_EQ(fmt::format("{:.2f}", std::complex(1, 2)), "(1.00+2.00i)"); EXPECT_EQ(fmt::format("{:8}", std::complex(1, 2)), " (1+2i)"); } #endif // FMT_STATIC_THOUSANDS_SEPARATOR