mirror of
				https://github.com/fmtlib/fmt.git
				synced 2025-11-03 23:51:41 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			434 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			434 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
// Formatting library for C++ - formatting library tests
 | 
						|
//
 | 
						|
// Copyright (c) 2012 - present, Victor Zverovich
 | 
						|
// All rights reserved.
 | 
						|
//
 | 
						|
// For the license information refer to format.h.
 | 
						|
 | 
						|
#include "fmt/xchar.h"
 | 
						|
 | 
						|
#include <complex>
 | 
						|
 | 
						|
#include "fmt/chrono.h"
 | 
						|
#include "fmt/color.h"
 | 
						|
#include "fmt/ostream.h"
 | 
						|
#include "fmt/ranges.h"
 | 
						|
#include "gtest/gtest.h"
 | 
						|
 | 
						|
using fmt::detail::max_value;
 | 
						|
 | 
						|
namespace test_ns {
 | 
						|
template <typename Char> class test_string {
 | 
						|
 private:
 | 
						|
  std::basic_string<Char> s_;
 | 
						|
 | 
						|
 public:
 | 
						|
  test_string(const Char* s) : s_(s) {}
 | 
						|
  const Char* data() const { return s_.data(); }
 | 
						|
  size_t length() const { return s_.size(); }
 | 
						|
  operator const Char*() const { return s_.c_str(); }
 | 
						|
};
 | 
						|
 | 
						|
template <typename Char>
 | 
						|
fmt::basic_string_view<Char> to_string_view(const test_string<Char>& s) {
 | 
						|
  return {s.data(), s.length()};
 | 
						|
}
 | 
						|
 | 
						|
struct non_string {};
 | 
						|
}  // namespace test_ns
 | 
						|
 | 
						|
template <typename T> class is_string_test : public testing::Test {};
 | 
						|
 | 
						|
using string_char_types = testing::Types<char, wchar_t, char16_t, char32_t>;
 | 
						|
TYPED_TEST_SUITE(is_string_test, string_char_types);
 | 
						|
 | 
						|
template <typename Char>
 | 
						|
struct derived_from_string_view : fmt::basic_string_view<Char> {};
 | 
						|
 | 
						|
TYPED_TEST(is_string_test, is_string) {
 | 
						|
  EXPECT_TRUE(fmt::detail::is_string<TypeParam*>::value);
 | 
						|
  EXPECT_TRUE(fmt::detail::is_string<const TypeParam*>::value);
 | 
						|
  EXPECT_TRUE(fmt::detail::is_string<TypeParam[2]>::value);
 | 
						|
  EXPECT_TRUE(fmt::detail::is_string<const TypeParam[2]>::value);
 | 
						|
  EXPECT_TRUE(fmt::detail::is_string<std::basic_string<TypeParam>>::value);
 | 
						|
  EXPECT_TRUE(fmt::detail::is_string<fmt::basic_string_view<TypeParam>>::value);
 | 
						|
  EXPECT_TRUE(
 | 
						|
      fmt::detail::is_string<derived_from_string_view<TypeParam>>::value);
 | 
						|
  using fmt_string_view = fmt::detail::std_string_view<TypeParam>;
 | 
						|
  EXPECT_TRUE(std::is_empty<fmt_string_view>::value !=
 | 
						|
              fmt::detail::is_string<fmt_string_view>::value);
 | 
						|
  EXPECT_TRUE(fmt::detail::is_string<test_ns::test_string<TypeParam>>::value);
 | 
						|
  EXPECT_FALSE(fmt::detail::is_string<test_ns::non_string>::value);
 | 
						|
}
 | 
						|
 | 
						|
// std::is_constructible is broken in MSVC until version 2015.
 | 
						|
#if !FMT_MSC_VER || FMT_MSC_VER >= 1900
 | 
						|
struct explicitly_convertible_to_wstring_view {
 | 
						|
  explicit operator fmt::wstring_view() const { return L"foo"; }
 | 
						|
};
 | 
						|
 | 
						|
TEST(xchar_test, format_explicitly_convertible_to_wstring_view) {
 | 
						|
  EXPECT_EQ(L"foo",
 | 
						|
            fmt::format(L"{}", explicitly_convertible_to_wstring_view()));
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
TEST(xchar_test, format) {
 | 
						|
  EXPECT_EQ(L"42", fmt::format(L"{}", 42));
 | 
						|
  EXPECT_EQ(L"4.2", fmt::format(L"{}", 4.2));
 | 
						|
  EXPECT_EQ(L"abc", fmt::format(L"{}", L"abc"));
 | 
						|
  EXPECT_EQ(L"z", fmt::format(L"{}", L'z'));
 | 
						|
  EXPECT_THROW(fmt::format(L"{:*\x343E}", 42), fmt::format_error);
 | 
						|
  EXPECT_EQ(L"true", fmt::format(L"{}", true));
 | 
						|
  EXPECT_EQ(L"a", fmt::format(L"{0}", 'a'));
 | 
						|
  EXPECT_EQ(L"a", fmt::format(L"{0}", L'a'));
 | 
						|
  EXPECT_EQ(L"Cyrillic letter \x42e",
 | 
						|
            fmt::format(L"Cyrillic letter {}", L'\x42e'));
 | 
						|
  EXPECT_EQ(L"abc1", fmt::format(L"{}c{}", L"ab", 1));
 | 
						|
}
 | 
						|
 | 
						|
TEST(xchar_test, is_formattable) {
 | 
						|
  static_assert(!fmt::is_formattable<const wchar_t*>::value, "");
 | 
						|
}
 | 
						|
 | 
						|
TEST(xchar_test, compile_time_string) {
 | 
						|
#if defined(FMT_USE_STRING_VIEW) && __cplusplus >= 201703L
 | 
						|
  EXPECT_EQ(L"42", fmt::format(FMT_STRING(std::wstring_view(L"{}")), 42));
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
#if __cplusplus > 201103L
 | 
						|
struct custom_char {
 | 
						|
  int value;
 | 
						|
  custom_char() = default;
 | 
						|
 | 
						|
  template <typename T>
 | 
						|
  constexpr custom_char(T val) : value(static_cast<int>(val)) {}
 | 
						|
 | 
						|
  operator int() const { return value; }
 | 
						|
};
 | 
						|
 | 
						|
int to_ascii(custom_char c) { return c; }
 | 
						|
 | 
						|
FMT_BEGIN_NAMESPACE
 | 
						|
template <> struct is_char<custom_char> : std::true_type {};
 | 
						|
FMT_END_NAMESPACE
 | 
						|
 | 
						|
TEST(xchar_test, format_custom_char) {
 | 
						|
  const custom_char format[] = {'{', '}', 0};
 | 
						|
  auto result = fmt::format(format, custom_char('x'));
 | 
						|
  EXPECT_EQ(result.size(), 1);
 | 
						|
  EXPECT_EQ(result[0], custom_char('x'));
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
// Convert a char8_t string to std::string. Otherwise GTest will insist on
 | 
						|
// inserting `char8_t` NTBS into a `char` stream which is disabled by P1423.
 | 
						|
template <typename S> std::string from_u8str(const S& str) {
 | 
						|
  return std::string(str.begin(), str.end());
 | 
						|
}
 | 
						|
 | 
						|
TEST(xchar_test, format_utf8_precision) {
 | 
						|
  using str_type = std::basic_string<fmt::detail::char8_type>;
 | 
						|
  auto format =
 | 
						|
      str_type(reinterpret_cast<const fmt::detail::char8_type*>(u8"{:.4}"));
 | 
						|
  auto str = str_type(reinterpret_cast<const fmt::detail::char8_type*>(
 | 
						|
      u8"caf\u00e9s"));  // cafés
 | 
						|
  auto result = fmt::format(format, str);
 | 
						|
  EXPECT_EQ(fmt::detail::compute_width(result), 4);
 | 
						|
  EXPECT_EQ(result.size(), 5);
 | 
						|
  EXPECT_EQ(from_u8str(result), from_u8str(str.substr(0, 5)));
 | 
						|
}
 | 
						|
 | 
						|
TEST(xchar_test, format_to) {
 | 
						|
  auto buf = std::vector<wchar_t>();
 | 
						|
  fmt::format_to(std::back_inserter(buf), L"{}{}", 42, L'\0');
 | 
						|
  EXPECT_STREQ(buf.data(), L"42");
 | 
						|
}
 | 
						|
 | 
						|
TEST(xchar_test, vformat_to) {
 | 
						|
  using wcontext = fmt::wformat_context;
 | 
						|
  fmt::basic_format_arg<wcontext> warg = fmt::detail::make_arg<wcontext>(42);
 | 
						|
  auto wargs = fmt::basic_format_args<wcontext>(&warg, 1);
 | 
						|
  auto w = std::wstring();
 | 
						|
  fmt::vformat_to(std::back_inserter(w), L"{}", wargs);
 | 
						|
  EXPECT_EQ(L"42", w);
 | 
						|
  w.clear();
 | 
						|
  fmt::vformat_to(std::back_inserter(w), FMT_STRING(L"{}"), wargs);
 | 
						|
  EXPECT_EQ(L"42", w);
 | 
						|
}
 | 
						|
 | 
						|
TEST(format_test, wide_format_to_n) {
 | 
						|
  wchar_t buffer[4];
 | 
						|
  buffer[3] = L'x';
 | 
						|
  auto result = fmt::format_to_n(buffer, 3, L"{}", 12345);
 | 
						|
  EXPECT_EQ(5u, result.size);
 | 
						|
  EXPECT_EQ(buffer + 3, result.out);
 | 
						|
  EXPECT_EQ(L"123x", fmt::wstring_view(buffer, 4));
 | 
						|
  buffer[0] = L'x';
 | 
						|
  buffer[1] = L'x';
 | 
						|
  buffer[2] = L'x';
 | 
						|
  result = fmt::format_to_n(buffer, 3, L"{}", L'A');
 | 
						|
  EXPECT_EQ(1u, result.size);
 | 
						|
  EXPECT_EQ(buffer + 1, result.out);
 | 
						|
  EXPECT_EQ(L"Axxx", fmt::wstring_view(buffer, 4));
 | 
						|
  result = fmt::format_to_n(buffer, 3, L"{}{} ", L'B', L'C');
 | 
						|
  EXPECT_EQ(3u, result.size);
 | 
						|
  EXPECT_EQ(buffer + 3, result.out);
 | 
						|
  EXPECT_EQ(L"BC x", fmt::wstring_view(buffer, 4));
 | 
						|
}
 | 
						|
 | 
						|
#if FMT_USE_USER_DEFINED_LITERALS
 | 
						|
TEST(xchar_test, format_udl) {
 | 
						|
  using namespace fmt::literals;
 | 
						|
  EXPECT_EQ(L"{}c{}"_format(L"ab", 1), fmt::format(L"{}c{}", L"ab", 1));
 | 
						|
}
 | 
						|
 | 
						|
TEST(xchar_test, named_arg_udl) {
 | 
						|
  using namespace fmt::literals;
 | 
						|
  auto udl_a =
 | 
						|
      fmt::format(L"{first}{second}{first}{third}", L"first"_a = L"abra",
 | 
						|
                  L"second"_a = L"cad", L"third"_a = 99);
 | 
						|
  EXPECT_EQ(
 | 
						|
      fmt::format(L"{first}{second}{first}{third}", fmt::arg(L"first", L"abra"),
 | 
						|
                  fmt::arg(L"second", L"cad"), fmt::arg(L"third", 99)),
 | 
						|
      udl_a);
 | 
						|
}
 | 
						|
#endif  // FMT_USE_USER_DEFINED_LITERALS
 | 
						|
 | 
						|
TEST(xchar_test, print) {
 | 
						|
  // Check that the wide print overload compiles.
 | 
						|
  if (fmt::detail::const_check(false)) fmt::print(L"test");
 | 
						|
}
 | 
						|
 | 
						|
TEST(xchar_test, join) {
 | 
						|
  int v[3] = {1, 2, 3};
 | 
						|
  EXPECT_EQ(fmt::format(L"({})", fmt::join(v, v + 3, L", ")), L"(1, 2, 3)");
 | 
						|
  auto t = std::tuple<wchar_t, int, float>('a', 1, 2.0f);
 | 
						|
  EXPECT_EQ(fmt::format(L"({})", fmt::join(t, L", ")), L"(a, 1, 2)");
 | 
						|
}
 | 
						|
 | 
						|
enum streamable_enum {};
 | 
						|
 | 
						|
std::wostream& operator<<(std::wostream& os, streamable_enum) {
 | 
						|
  return os << L"streamable_enum";
 | 
						|
}
 | 
						|
 | 
						|
enum unstreamable_enum {};
 | 
						|
 | 
						|
TEST(xchar_test, enum) {
 | 
						|
  EXPECT_EQ(L"streamable_enum", fmt::format(L"{}", streamable_enum()));
 | 
						|
  EXPECT_EQ(L"0", fmt::format(L"{}", unstreamable_enum()));
 | 
						|
}
 | 
						|
 | 
						|
TEST(xchar_test, sign_not_truncated) {
 | 
						|
  wchar_t format_str[] = {
 | 
						|
      L'{', L':',
 | 
						|
      '+' | static_cast<wchar_t>(1 << fmt::detail::num_bits<char>()), L'}', 0};
 | 
						|
  EXPECT_THROW(fmt::format(format_str, 42), fmt::format_error);
 | 
						|
}
 | 
						|
 | 
						|
namespace fake_qt {
 | 
						|
class QString {
 | 
						|
 public:
 | 
						|
  QString(const wchar_t* s) : s_(s) {}
 | 
						|
  const wchar_t* utf16() const FMT_NOEXCEPT { return s_.data(); }
 | 
						|
  int size() const FMT_NOEXCEPT { return static_cast<int>(s_.size()); }
 | 
						|
 | 
						|
 private:
 | 
						|
  std::wstring s_;
 | 
						|
};
 | 
						|
 | 
						|
fmt::basic_string_view<wchar_t> to_string_view(const QString& s) FMT_NOEXCEPT {
 | 
						|
  return {s.utf16(), static_cast<size_t>(s.size())};
 | 
						|
}
 | 
						|
}  // namespace fake_qt
 | 
						|
 | 
						|
TEST(format_test, format_foreign_strings) {
 | 
						|
  using fake_qt::QString;
 | 
						|
  EXPECT_EQ(fmt::format(QString(L"{}"), 42), L"42");
 | 
						|
  EXPECT_EQ(fmt::format(QString(L"{}"), QString(L"42")), L"42");
 | 
						|
}
 | 
						|
 | 
						|
TEST(xchar_test, chrono) {
 | 
						|
  auto tm = std::tm();
 | 
						|
  tm.tm_year = 116;
 | 
						|
  tm.tm_mon = 3;
 | 
						|
  tm.tm_mday = 25;
 | 
						|
  tm.tm_hour = 11;
 | 
						|
  tm.tm_min = 22;
 | 
						|
  tm.tm_sec = 33;
 | 
						|
  EXPECT_EQ(fmt::format("The date is {:%Y-%m-%d %H:%M:%S}.", tm),
 | 
						|
            "The date is 2016-04-25 11:22:33.");
 | 
						|
  EXPECT_EQ(L"42s", fmt::format(L"{}", std::chrono::seconds(42)));
 | 
						|
}
 | 
						|
 | 
						|
TEST(xchar_test, color) {
 | 
						|
  EXPECT_EQ(fmt::format(fg(fmt::rgb(255, 20, 30)), L"rgb(255,20,30) wide"),
 | 
						|
            L"\x1b[38;2;255;020;030mrgb(255,20,30) wide\x1b[0m");
 | 
						|
}
 | 
						|
 | 
						|
TEST(xchar_test, ostream) {
 | 
						|
  std::wostringstream wos;
 | 
						|
  fmt::print(wos, L"Don't {}!", L"panic");
 | 
						|
  EXPECT_EQ(L"Don't panic!", wos.str());
 | 
						|
}
 | 
						|
 | 
						|
TEST(xchar_test, to_wstring) { EXPECT_EQ(L"42", fmt::to_wstring(42)); }
 | 
						|
 | 
						|
#ifndef FMT_STATIC_THOUSANDS_SEPARATOR
 | 
						|
template <typename Char> struct numpunct : std::numpunct<Char> {
 | 
						|
 protected:
 | 
						|
  Char do_decimal_point() const override { return '?'; }
 | 
						|
  std::string do_grouping() const override { return "\03"; }
 | 
						|
  Char do_thousands_sep() const override { return '~'; }
 | 
						|
};
 | 
						|
 | 
						|
template <typename Char> struct no_grouping : std::numpunct<Char> {
 | 
						|
 protected:
 | 
						|
  Char do_decimal_point() const override { return '.'; }
 | 
						|
  std::string do_grouping() const override { return ""; }
 | 
						|
  Char do_thousands_sep() const override { return ','; }
 | 
						|
};
 | 
						|
 | 
						|
template <typename Char> struct special_grouping : std::numpunct<Char> {
 | 
						|
 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 <typename Char> struct small_grouping : std::numpunct<Char> {
 | 
						|
 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, localized_double) {
 | 
						|
  auto loc = std::locale(std::locale(), new numpunct<char>());
 | 
						|
  EXPECT_EQ("1?23", fmt::format(loc, "{:L}", 1.23));
 | 
						|
  EXPECT_EQ("1?230000", fmt::format(loc, "{:Lf}", 1.23));
 | 
						|
  EXPECT_EQ("1~234?5", fmt::format(loc, "{:L}", 1234.5));
 | 
						|
  EXPECT_EQ("12~000", fmt::format(loc, "{:L}", 12000.0));
 | 
						|
}
 | 
						|
 | 
						|
TEST(locale_test, format) {
 | 
						|
  auto loc = std::locale(std::locale(), new numpunct<char>());
 | 
						|
  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<fmt::format_context, int> 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<char>());
 | 
						|
  EXPECT_EQ("1234567", fmt::format(no_grouping_loc, "{:L}", 1234567));
 | 
						|
 | 
						|
  auto special_grouping_loc =
 | 
						|
      std::locale(std::locale(), new special_grouping<char>());
 | 
						|
  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<char>());
 | 
						|
  EXPECT_EQ("4,2,9,4,9,6,7,2,9,5",
 | 
						|
            fmt::format(small_grouping_loc, "{:L}", max_value<uint32_t>()));
 | 
						|
}
 | 
						|
 | 
						|
TEST(locale_test, format_detault_align) {
 | 
						|
  auto loc = std::locale({}, new special_grouping<char>());
 | 
						|
  EXPECT_EQ("  12,345", fmt::format(loc, "{:8L}", 12345));
 | 
						|
}
 | 
						|
 | 
						|
TEST(locale_test, format_plus) {
 | 
						|
  auto loc = std::locale({}, new special_grouping<char>());
 | 
						|
  EXPECT_EQ("+100", fmt::format(loc, "{:+L}", 100));
 | 
						|
}
 | 
						|
 | 
						|
TEST(locale_test, wformat) {
 | 
						|
  auto loc = std::locale(std::locale(), new numpunct<wchar_t>());
 | 
						|
  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<wchar_t>;
 | 
						|
  fmt::format_arg_store<wcontext, int> as{1234567};
 | 
						|
  EXPECT_EQ(L"1~234~567",
 | 
						|
            fmt::vformat(loc, L"{:L}", fmt::basic_format_args<wcontext>(as)));
 | 
						|
  EXPECT_EQ(L"1234567", fmt::format(std::locale("C"), L"{:L}", 1234567));
 | 
						|
 | 
						|
  auto no_grouping_loc = std::locale(std::locale(), new no_grouping<wchar_t>());
 | 
						|
  EXPECT_EQ(L"1234567", fmt::format(no_grouping_loc, L"{:L}", 1234567));
 | 
						|
 | 
						|
  auto special_grouping_loc =
 | 
						|
      std::locale(std::locale(), new special_grouping<wchar_t>());
 | 
						|
  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<wchar_t>());
 | 
						|
  EXPECT_EQ(L"4,2,9,4,9,6,7,2,9,5",
 | 
						|
            fmt::format(small_grouping_loc, L"{:L}", max_value<uint32_t>()));
 | 
						|
}
 | 
						|
 | 
						|
TEST(locale_test, double_formatter) {
 | 
						|
  auto loc = std::locale(std::locale(), new special_grouping<char>());
 | 
						|
  auto f = fmt::formatter<int>();
 | 
						|
  auto parse_ctx = fmt::format_parse_context("L");
 | 
						|
  f.parse(parse_ctx);
 | 
						|
  char buf[10] = {};
 | 
						|
  fmt::basic_format_context<char*, char> format_ctx(
 | 
						|
      buf, {}, fmt::detail::locale_ref(loc));
 | 
						|
  *f.format(12345, format_ctx) = 0;
 | 
						|
  EXPECT_STREQ("12,345", buf);
 | 
						|
}
 | 
						|
 | 
						|
FMT_BEGIN_NAMESPACE
 | 
						|
template <class charT> struct formatter<std::complex<double>, charT> {
 | 
						|
 private:
 | 
						|
  detail::dynamic_format_specs<char> specs_;
 | 
						|
 | 
						|
 public:
 | 
						|
  FMT_CONSTEXPR typename basic_format_parse_context<charT>::iterator parse(
 | 
						|
      basic_format_parse_context<charT>& ctx) {
 | 
						|
    using handler_type =
 | 
						|
        detail::dynamic_specs_handler<basic_format_parse_context<charT>>;
 | 
						|
    detail::specs_checker<handler_type> 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 <class FormatContext>
 | 
						|
  typename FormatContext::iterator format(const std::complex<double>& c,
 | 
						|
                                          FormatContext& ctx) {
 | 
						|
    detail::handle_dynamic_spec<detail::precision_checker>(
 | 
						|
        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<std::locale>(),
 | 
						|
                            fmt::runtime("{:" + specs + "}"), c.real());
 | 
						|
    auto imag = fmt::format(ctx.locale().template get<std::locale>(),
 | 
						|
                            fmt::runtime("{:" + 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<double>(1, 2));
 | 
						|
  EXPECT_EQ(s, "(1+2i)");
 | 
						|
  EXPECT_EQ(fmt::format("{:.2f}", std::complex<double>(1, 2)), "(1.00+2.00i)");
 | 
						|
  EXPECT_EQ(fmt::format("{:8}", std::complex<double>(1, 2)), "  (1+2i)");
 | 
						|
}
 | 
						|
 | 
						|
#endif  // FMT_STATIC_THOUSANDS_SEPARATOR
 |