Mercurial > minori
comparison dep/fmt/test/xchar-test.cc @ 343:1faa72660932
*: transfer back to cmake from autotools
autotools just made lots of things more complicated than
they should have and many things broke (i.e. translations)
| author | Paper <paper@paper.us.eu.org> | 
|---|---|
| date | Thu, 20 Jun 2024 05:56:06 -0400 | 
| parents | |
| children | 
   comparison
  equal
  deleted
  inserted
  replaced
| 342:adb79bdde329 | 343:1faa72660932 | 
|---|---|
| 1 // Formatting library for C++ - formatting library tests | |
| 2 // | |
| 3 // Copyright (c) 2012 - present, Victor Zverovich | |
| 4 // All rights reserved. | |
| 5 // | |
| 6 // For the license information refer to format.h. | |
| 7 | |
| 8 #include "fmt/xchar.h" | |
| 9 | |
| 10 #include <algorithm> | |
| 11 #include <complex> | |
| 12 #include <cwchar> | |
| 13 #include <vector> | |
| 14 | |
| 15 #include "fmt/chrono.h" | |
| 16 #include "fmt/color.h" | |
| 17 #include "fmt/ostream.h" | |
| 18 #include "fmt/ranges.h" | |
| 19 #include "fmt/std.h" | |
| 20 #include "gtest-extra.h" // Contains | |
| 21 #include "util.h" // get_locale | |
| 22 | |
| 23 using fmt::detail::max_value; | |
| 24 using testing::Contains; | |
| 25 | |
| 26 #if defined(__MINGW32__) && !defined(_UCRT) | |
| 27 // Only C89 conversion specifiers when using MSVCRT instead of UCRT | |
| 28 # define FMT_HAS_C99_STRFTIME 0 | |
| 29 #else | |
| 30 # define FMT_HAS_C99_STRFTIME 1 | |
| 31 #endif | |
| 32 | |
| 33 struct non_string {}; | |
| 34 | |
| 35 template <typename T> class is_string_test : public testing::Test {}; | |
| 36 | |
| 37 using string_char_types = testing::Types<char, wchar_t, char16_t, char32_t>; | |
| 38 TYPED_TEST_SUITE(is_string_test, string_char_types); | |
| 39 | |
| 40 template <typename Char> | |
| 41 struct derived_from_string_view : fmt::basic_string_view<Char> {}; | |
| 42 | |
| 43 TYPED_TEST(is_string_test, is_string) { | |
| 44 EXPECT_TRUE(fmt::detail::is_string<TypeParam*>::value); | |
| 45 EXPECT_TRUE(fmt::detail::is_string<const TypeParam*>::value); | |
| 46 EXPECT_TRUE(fmt::detail::is_string<TypeParam[2]>::value); | |
| 47 EXPECT_TRUE(fmt::detail::is_string<const TypeParam[2]>::value); | |
| 48 EXPECT_TRUE(fmt::detail::is_string<std::basic_string<TypeParam>>::value); | |
| 49 EXPECT_TRUE(fmt::detail::is_string<fmt::basic_string_view<TypeParam>>::value); | |
| 50 EXPECT_TRUE( | |
| 51 fmt::detail::is_string<derived_from_string_view<TypeParam>>::value); | |
| 52 using fmt_string_view = fmt::detail::std_string_view<TypeParam>; | |
| 53 EXPECT_TRUE(std::is_empty<fmt_string_view>::value != | |
| 54 fmt::detail::is_string<fmt_string_view>::value); | |
| 55 EXPECT_FALSE(fmt::detail::is_string<non_string>::value); | |
| 56 } | |
| 57 | |
| 58 // std::is_constructible is broken in MSVC until version 2015. | |
| 59 #if !FMT_MSC_VERSION || FMT_MSC_VERSION >= 1900 | |
| 60 struct explicitly_convertible_to_wstring_view { | |
| 61 explicit operator fmt::wstring_view() const { return L"foo"; } | |
| 62 }; | |
| 63 | |
| 64 TEST(xchar_test, format_explicitly_convertible_to_wstring_view) { | |
| 65 // Types explicitly convertible to wstring_view are not formattable by | |
| 66 // default because it may introduce ODR violations. | |
| 67 static_assert( | |
| 68 !fmt::is_formattable<explicitly_convertible_to_wstring_view>::value, ""); | |
| 69 } | |
| 70 #endif | |
| 71 | |
| 72 TEST(xchar_test, format) { | |
| 73 EXPECT_EQ(L"42", fmt::format(L"{}", 42)); | |
| 74 EXPECT_EQ(L"4.2", fmt::format(L"{}", 4.2)); | |
| 75 EXPECT_EQ(L"abc", fmt::format(L"{}", L"abc")); | |
| 76 EXPECT_EQ(L"z", fmt::format(L"{}", L'z')); | |
| 77 EXPECT_THROW(fmt::format(fmt::runtime(L"{:*\x343E}"), 42), fmt::format_error); | |
| 78 EXPECT_EQ(L"true", fmt::format(L"{}", true)); | |
| 79 EXPECT_EQ(L"a", fmt::format(L"{0}", 'a')); | |
| 80 EXPECT_EQ(L"a", fmt::format(L"{0}", L'a')); | |
| 81 EXPECT_EQ(L"Cyrillic letter \x42e", | |
| 82 fmt::format(L"Cyrillic letter {}", L'\x42e')); | |
| 83 EXPECT_EQ(L"abc1", fmt::format(L"{}c{}", L"ab", 1)); | |
| 84 } | |
| 85 | |
| 86 TEST(xchar_test, is_formattable) { | |
| 87 static_assert(!fmt::is_formattable<const wchar_t*>::value, ""); | |
| 88 } | |
| 89 | |
| 90 TEST(xchar_test, compile_time_string) { | |
| 91 EXPECT_EQ(fmt::format(fmt::wformat_string<int>(L"{}"), 42), L"42"); | |
| 92 #if defined(FMT_USE_STRING_VIEW) && FMT_CPLUSPLUS >= 201703L | |
| 93 EXPECT_EQ(fmt::format(FMT_STRING(std::wstring_view(L"{}")), 42), L"42"); | |
| 94 #endif | |
| 95 } | |
| 96 | |
| 97 #if FMT_CPLUSPLUS > 201103L | |
| 98 struct custom_char { | |
| 99 int value; | |
| 100 custom_char() = default; | |
| 101 | |
| 102 template <typename T> | |
| 103 constexpr custom_char(T val) : value(static_cast<int>(val)) {} | |
| 104 | |
| 105 constexpr operator char() const { | |
| 106 return value <= 0xff ? static_cast<char>(value) : '\0'; | |
| 107 } | |
| 108 constexpr bool operator<(custom_char c) const { return value < c.value; } | |
| 109 }; | |
| 110 | |
| 111 namespace std { | |
| 112 | |
| 113 template <> struct char_traits<custom_char> { | |
| 114 using char_type = custom_char; | |
| 115 using int_type = int; | |
| 116 using off_type = streamoff; | |
| 117 using pos_type = streampos; | |
| 118 using state_type = mbstate_t; | |
| 119 | |
| 120 static constexpr void assign(char_type& r, const char_type& a) { r = a; } | |
| 121 static constexpr bool eq(char_type a, char_type b) { return a == b; } | |
| 122 static constexpr bool lt(char_type a, char_type b) { return a < b; } | |
| 123 static FMT_CONSTEXPR int compare(const char_type* s1, const char_type* s2, | |
| 124 size_t count) { | |
| 125 for (; count; count--, s1++, s2++) { | |
| 126 if (lt(*s1, *s2)) return -1; | |
| 127 if (lt(*s2, *s1)) return 1; | |
| 128 } | |
| 129 return 0; | |
| 130 } | |
| 131 static FMT_CONSTEXPR size_t length(const char_type* s) { | |
| 132 size_t count = 0; | |
| 133 while (!eq(*s++, custom_char(0))) count++; | |
| 134 return count; | |
| 135 } | |
| 136 static const char_type* find(const char_type*, size_t, const char_type&); | |
| 137 static FMT_CONSTEXPR char_type* move(char_type* dest, const char_type* src, | |
| 138 size_t count) { | |
| 139 if (count == 0) return dest; | |
| 140 char_type* ret = dest; | |
| 141 if (src < dest) { | |
| 142 dest += count; | |
| 143 src += count; | |
| 144 for (; count; count--) assign(*--dest, *--src); | |
| 145 } else if (src > dest) | |
| 146 copy(dest, src, count); | |
| 147 return ret; | |
| 148 } | |
| 149 static FMT_CONSTEXPR char_type* copy(char_type* dest, const char_type* src, | |
| 150 size_t count) { | |
| 151 char_type* ret = dest; | |
| 152 for (; count; count--) assign(*dest++, *src++); | |
| 153 return ret; | |
| 154 } | |
| 155 static FMT_CONSTEXPR char_type* assign(char_type* dest, std::size_t count, | |
| 156 char_type a) { | |
| 157 char_type* ret = dest; | |
| 158 for (; count; count--) assign(*dest++, a); | |
| 159 return ret; | |
| 160 } | |
| 161 static int_type not_eof(int_type); | |
| 162 static char_type to_char_type(int_type); | |
| 163 static int_type to_int_type(char_type); | |
| 164 static bool eq_int_type(int_type, int_type); | |
| 165 static int_type eof(); | |
| 166 }; | |
| 167 | |
| 168 } // namespace std | |
| 169 | |
| 170 auto to_ascii(custom_char c) -> char { return c; } | |
| 171 | |
| 172 FMT_BEGIN_NAMESPACE | |
| 173 template <> struct is_char<custom_char> : std::true_type {}; | |
| 174 FMT_END_NAMESPACE | |
| 175 | |
| 176 TEST(xchar_test, format_custom_char) { | |
| 177 const custom_char format[] = {'{', '}', 0}; | |
| 178 auto result = fmt::format(format, custom_char('x')); | |
| 179 EXPECT_EQ(result.size(), 1); | |
| 180 EXPECT_EQ(result[0], custom_char('x')); | |
| 181 } | |
| 182 #endif | |
| 183 | |
| 184 // Convert a char8_t string to std::string. Otherwise GTest will insist on | |
| 185 // inserting `char8_t` NTBS into a `char` stream which is disabled by P1423. | |
| 186 template <typename S> std::string from_u8str(const S& str) { | |
| 187 return std::string(str.begin(), str.end()); | |
| 188 } | |
| 189 | |
| 190 TEST(xchar_test, format_to) { | |
| 191 auto buf = std::vector<wchar_t>(); | |
| 192 fmt::format_to(std::back_inserter(buf), L"{}{}", 42, L'\0'); | |
| 193 EXPECT_STREQ(buf.data(), L"42"); | |
| 194 } | |
| 195 | |
| 196 TEST(xchar_test, vformat_to) { | |
| 197 auto args = fmt::make_wformat_args(42); | |
| 198 auto w = std::wstring(); | |
| 199 fmt::vformat_to(std::back_inserter(w), L"{}", args); | |
| 200 EXPECT_EQ(L"42", w); | |
| 201 } | |
| 202 | |
| 203 namespace test { | |
| 204 struct struct_as_wstring_view {}; | |
| 205 auto format_as(struct_as_wstring_view) -> fmt::wstring_view { return L"foo"; } | |
| 206 } // namespace test | |
| 207 | |
| 208 TEST(xchar_test, format_as) { | |
| 209 EXPECT_EQ(fmt::format(L"{}", test::struct_as_wstring_view()), L"foo"); | |
| 210 } | |
| 211 | |
| 212 TEST(format_test, wide_format_to_n) { | |
| 213 wchar_t buffer[4]; | |
| 214 buffer[3] = L'x'; | |
| 215 auto result = fmt::format_to_n(buffer, 3, L"{}", 12345); | |
| 216 EXPECT_EQ(5u, result.size); | |
| 217 EXPECT_EQ(buffer + 3, result.out); | |
| 218 EXPECT_EQ(L"123x", fmt::wstring_view(buffer, 4)); | |
| 219 buffer[0] = L'x'; | |
| 220 buffer[1] = L'x'; | |
| 221 buffer[2] = L'x'; | |
| 222 result = fmt::format_to_n(buffer, 3, L"{}", L'A'); | |
| 223 EXPECT_EQ(1u, result.size); | |
| 224 EXPECT_EQ(buffer + 1, result.out); | |
| 225 EXPECT_EQ(L"Axxx", fmt::wstring_view(buffer, 4)); | |
| 226 result = fmt::format_to_n(buffer, 3, L"{}{} ", L'B', L'C'); | |
| 227 EXPECT_EQ(3u, result.size); | |
| 228 EXPECT_EQ(buffer + 3, result.out); | |
| 229 EXPECT_EQ(L"BC x", fmt::wstring_view(buffer, 4)); | |
| 230 } | |
| 231 | |
| 232 #if FMT_USE_USER_DEFINED_LITERALS | |
| 233 TEST(xchar_test, named_arg_udl) { | |
| 234 using namespace fmt::literals; | |
| 235 auto udl_a = | |
| 236 fmt::format(L"{first}{second}{first}{third}", L"first"_a = L"abra", | |
| 237 L"second"_a = L"cad", L"third"_a = 99); | |
| 238 EXPECT_EQ( | |
| 239 fmt::format(L"{first}{second}{first}{third}", fmt::arg(L"first", L"abra"), | |
| 240 fmt::arg(L"second", L"cad"), fmt::arg(L"third", 99)), | |
| 241 udl_a); | |
| 242 } | |
| 243 #endif // FMT_USE_USER_DEFINED_LITERALS | |
| 244 | |
| 245 TEST(xchar_test, print) { | |
| 246 // Check that the wide print overload compiles. | |
| 247 if (fmt::detail::const_check(false)) { | |
| 248 fmt::print(L"test"); | |
| 249 fmt::println(L"test"); | |
| 250 } | |
| 251 } | |
| 252 | |
| 253 TEST(xchar_test, join) { | |
| 254 int v[3] = {1, 2, 3}; | |
| 255 EXPECT_EQ(fmt::format(L"({})", fmt::join(v, v + 3, L", ")), L"(1, 2, 3)"); | |
| 256 auto t = std::tuple<wchar_t, int, float>('a', 1, 2.0f); | |
| 257 EXPECT_EQ(fmt::format(L"({})", fmt::join(t, L", ")), L"(a, 1, 2)"); | |
| 258 } | |
| 259 | |
| 260 enum streamable_enum {}; | |
| 261 | |
| 262 std::wostream& operator<<(std::wostream& os, streamable_enum) { | |
| 263 return os << L"streamable_enum"; | |
| 264 } | |
| 265 | |
| 266 namespace fmt { | |
| 267 template <> | |
| 268 struct formatter<streamable_enum, wchar_t> : basic_ostream_formatter<wchar_t> { | |
| 269 }; | |
| 270 } // namespace fmt | |
| 271 | |
| 272 enum unstreamable_enum {}; | |
| 273 auto format_as(unstreamable_enum e) -> int { return e; } | |
| 274 | |
| 275 TEST(xchar_test, enum) { | |
| 276 EXPECT_EQ(L"streamable_enum", fmt::format(L"{}", streamable_enum())); | |
| 277 EXPECT_EQ(L"0", fmt::format(L"{}", unstreamable_enum())); | |
| 278 } | |
| 279 | |
| 280 struct streamable_and_unformattable {}; | |
| 281 | |
| 282 auto operator<<(std::wostream& os, streamable_and_unformattable) | |
| 283 -> std::wostream& { | |
| 284 return os << L"foo"; | |
| 285 } | |
| 286 | |
| 287 TEST(xchar_test, streamed) { | |
| 288 EXPECT_FALSE(fmt::is_formattable<streamable_and_unformattable>()); | |
| 289 EXPECT_EQ(fmt::format(L"{}", fmt::streamed(streamable_and_unformattable())), | |
| 290 L"foo"); | |
| 291 } | |
| 292 | |
| 293 TEST(xchar_test, sign_not_truncated) { | |
| 294 wchar_t format_str[] = { | |
| 295 L'{', L':', | |
| 296 '+' | static_cast<wchar_t>(1 << fmt::detail::num_bits<char>()), L'}', 0}; | |
| 297 EXPECT_THROW(fmt::format(fmt::runtime(format_str), 42), fmt::format_error); | |
| 298 } | |
| 299 | |
| 300 TEST(xchar_test, chrono) { | |
| 301 auto tm = std::tm(); | |
| 302 tm.tm_year = 116; | |
| 303 tm.tm_mon = 3; | |
| 304 tm.tm_mday = 25; | |
| 305 tm.tm_hour = 11; | |
| 306 tm.tm_min = 22; | |
| 307 tm.tm_sec = 33; | |
| 308 EXPECT_EQ(fmt::format("The date is {:%Y-%m-%d %H:%M:%S}.", tm), | |
| 309 "The date is 2016-04-25 11:22:33."); | |
| 310 EXPECT_EQ(L"42s", fmt::format(L"{}", std::chrono::seconds(42))); | |
| 311 EXPECT_EQ(fmt::format(L"{:%F}", tm), L"2016-04-25"); | |
| 312 EXPECT_EQ(fmt::format(L"{:%T}", tm), L"11:22:33"); | |
| 313 } | |
| 314 | |
| 315 std::wstring system_wcsftime(const std::wstring& format, const std::tm* timeptr, | |
| 316 std::locale* locptr = nullptr) { | |
| 317 auto loc = locptr ? *locptr : std::locale::classic(); | |
| 318 auto& facet = std::use_facet<std::time_put<wchar_t>>(loc); | |
| 319 std::wostringstream os; | |
| 320 os.imbue(loc); | |
| 321 facet.put(os, os, L' ', timeptr, format.c_str(), | |
| 322 format.c_str() + format.size()); | |
| 323 #ifdef _WIN32 | |
| 324 // Workaround a bug in older versions of Universal CRT. | |
| 325 auto str = os.str(); | |
| 326 if (str == L"-0000") str = L"+0000"; | |
| 327 return str; | |
| 328 #else | |
| 329 return os.str(); | |
| 330 #endif | |
| 331 } | |
| 332 | |
| 333 TEST(chrono_test_wchar, time_point) { | |
| 334 auto t1 = std::chrono::time_point_cast<std::chrono::seconds>( | |
| 335 std::chrono::system_clock::now()); | |
| 336 | |
| 337 std::vector<std::wstring> spec_list = { | |
| 338 L"%%", L"%n", L"%t", L"%Y", L"%EY", L"%y", L"%Oy", L"%Ey", L"%C", | |
| 339 L"%EC", L"%G", L"%g", L"%b", L"%h", L"%B", L"%m", L"%Om", L"%U", | |
| 340 L"%OU", L"%W", L"%OW", L"%V", L"%OV", L"%j", L"%d", L"%Od", L"%e", | |
| 341 L"%Oe", L"%a", L"%A", L"%w", L"%Ow", L"%u", L"%Ou", L"%H", L"%OH", | |
| 342 L"%I", L"%OI", L"%M", L"%OM", L"%S", L"%OS", L"%x", L"%Ex", L"%X", | |
| 343 L"%EX", L"%D", L"%F", L"%R", L"%T", L"%p"}; | |
| 344 #ifndef _WIN32 | |
| 345 // Disabled on Windows, because these formats is not consistent among | |
| 346 // platforms. | |
| 347 spec_list.insert(spec_list.end(), {L"%c", L"%Ec", L"%r"}); | |
| 348 #elif !FMT_HAS_C99_STRFTIME | |
| 349 // Only C89 conversion specifiers when using MSVCRT instead of UCRT | |
| 350 spec_list = {L"%%", L"%Y", L"%y", L"%b", L"%B", L"%m", L"%U", | |
| 351 L"%W", L"%j", L"%d", L"%a", L"%A", L"%w", L"%H", | |
| 352 L"%I", L"%M", L"%S", L"%x", L"%X", L"%p"}; | |
| 353 #endif | |
| 354 spec_list.push_back(L"%Y-%m-%d %H:%M:%S"); | |
| 355 | |
| 356 for (const auto& spec : spec_list) { | |
| 357 auto t = std::chrono::system_clock::to_time_t(t1); | |
| 358 auto tm = *std::gmtime(&t); | |
| 359 | |
| 360 auto sys_output = system_wcsftime(spec, &tm); | |
| 361 | |
| 362 auto fmt_spec = fmt::format(L"{{:{}}}", spec); | |
| 363 EXPECT_EQ(sys_output, fmt::format(fmt::runtime(fmt_spec), t1)); | |
| 364 EXPECT_EQ(sys_output, fmt::format(fmt::runtime(fmt_spec), tm)); | |
| 365 } | |
| 366 | |
| 367 // Timezone formatters tests makes sense for localtime. | |
| 368 #if FMT_HAS_C99_STRFTIME | |
| 369 spec_list = {L"%z", L"%Z"}; | |
| 370 #else | |
| 371 spec_list = {L"%Z"}; | |
| 372 #endif | |
| 373 for (const auto& spec : spec_list) { | |
| 374 auto t = std::chrono::system_clock::to_time_t(t1); | |
| 375 auto tm = *std::localtime(&t); | |
| 376 | |
| 377 auto sys_output = system_wcsftime(spec, &tm); | |
| 378 | |
| 379 auto fmt_spec = fmt::format(L"{{:{}}}", spec); | |
| 380 EXPECT_EQ(sys_output, fmt::format(fmt::runtime(fmt_spec), tm)); | |
| 381 | |
| 382 if (spec == L"%z") { | |
| 383 sys_output.insert(sys_output.end() - 2, 1, L':'); | |
| 384 EXPECT_EQ(sys_output, fmt::format(L"{:%Ez}", tm)); | |
| 385 EXPECT_EQ(sys_output, fmt::format(L"{:%Oz}", tm)); | |
| 386 } | |
| 387 } | |
| 388 | |
| 389 // Separate tests for UTC, since std::time_put can use local time and ignoring | |
| 390 // the timezone in std::tm (if it presents on platform). | |
| 391 if (fmt::detail::has_member_data_tm_zone<std::tm>::value) { | |
| 392 auto t = std::chrono::system_clock::to_time_t(t1); | |
| 393 auto tm = *std::gmtime(&t); | |
| 394 | |
| 395 std::vector<std::wstring> tz_names = {L"GMT", L"UTC"}; | |
| 396 EXPECT_THAT(tz_names, Contains(fmt::format(L"{:%Z}", t1))); | |
| 397 EXPECT_THAT(tz_names, Contains(fmt::format(L"{:%Z}", tm))); | |
| 398 } | |
| 399 | |
| 400 if (fmt::detail::has_member_data_tm_gmtoff<std::tm>::value) { | |
| 401 auto t = std::chrono::system_clock::to_time_t(t1); | |
| 402 auto tm = *std::gmtime(&t); | |
| 403 | |
| 404 EXPECT_EQ(L"+0000", fmt::format(L"{:%z}", t1)); | |
| 405 EXPECT_EQ(L"+0000", fmt::format(L"{:%z}", tm)); | |
| 406 | |
| 407 EXPECT_EQ(L"+00:00", fmt::format(L"{:%Ez}", t1)); | |
| 408 EXPECT_EQ(L"+00:00", fmt::format(L"{:%Ez}", tm)); | |
| 409 | |
| 410 EXPECT_EQ(L"+00:00", fmt::format(L"{:%Oz}", t1)); | |
| 411 EXPECT_EQ(L"+00:00", fmt::format(L"{:%Oz}", tm)); | |
| 412 } | |
| 413 } | |
| 414 | |
| 415 TEST(xchar_test, color) { | |
| 416 EXPECT_EQ(fmt::format(fg(fmt::rgb(255, 20, 30)), L"rgb(255,20,30) wide"), | |
| 417 L"\x1b[38;2;255;020;030mrgb(255,20,30) wide\x1b[0m"); | |
| 418 } | |
| 419 | |
| 420 TEST(xchar_test, ostream) { | |
| 421 #if !FMT_GCC_VERSION || FMT_GCC_VERSION >= 409 | |
| 422 { | |
| 423 std::wostringstream wos; | |
| 424 fmt::print(wos, L"Don't {}!", L"panic"); | |
| 425 EXPECT_EQ(wos.str(), L"Don't panic!"); | |
| 426 } | |
| 427 | |
| 428 { | |
| 429 std::wostringstream wos; | |
| 430 fmt::println(wos, L"Don't {}!", L"panic"); | |
| 431 EXPECT_EQ(wos.str(), L"Don't panic!\n"); | |
| 432 } | |
| 433 #endif | |
| 434 } | |
| 435 | |
| 436 TEST(xchar_test, format_map) { | |
| 437 auto m = std::map<std::wstring, int>{{L"one", 1}, {L"t\"wo", 2}}; | |
| 438 EXPECT_EQ(fmt::format(L"{}", m), L"{\"one\": 1, \"t\\\"wo\": 2}"); | |
| 439 } | |
| 440 | |
| 441 TEST(xchar_test, escape_string) { | |
| 442 using vec = std::vector<std::wstring>; | |
| 443 EXPECT_EQ(fmt::format(L"{}", vec{L"\n\r\t\"\\"}), L"[\"\\n\\r\\t\\\"\\\\\"]"); | |
| 444 EXPECT_EQ(fmt::format(L"{}", vec{L"понедельник"}), L"[\"понедельник\"]"); | |
| 445 } | |
| 446 | |
| 447 TEST(xchar_test, to_wstring) { EXPECT_EQ(L"42", fmt::to_wstring(42)); } | |
| 448 | |
| 449 #ifndef FMT_STATIC_THOUSANDS_SEPARATOR | |
| 450 | |
| 451 template <typename Char> struct numpunct : std::numpunct<Char> { | |
| 452 protected: | |
| 453 Char do_decimal_point() const override { return '?'; } | |
| 454 std::string do_grouping() const override { return "\03"; } | |
| 455 Char do_thousands_sep() const override { return '~'; } | |
| 456 }; | |
| 457 | |
| 458 template <typename Char> struct no_grouping : std::numpunct<Char> { | |
| 459 protected: | |
| 460 Char do_decimal_point() const override { return '.'; } | |
| 461 std::string do_grouping() const override { return ""; } | |
| 462 Char do_thousands_sep() const override { return ','; } | |
| 463 }; | |
| 464 | |
| 465 template <typename Char> struct special_grouping : std::numpunct<Char> { | |
| 466 protected: | |
| 467 Char do_decimal_point() const override { return '.'; } | |
| 468 std::string do_grouping() const override { return "\03\02"; } | |
| 469 Char do_thousands_sep() const override { return ','; } | |
| 470 }; | |
| 471 | |
| 472 template <typename Char> struct small_grouping : std::numpunct<Char> { | |
| 473 protected: | |
| 474 Char do_decimal_point() const override { return '.'; } | |
| 475 std::string do_grouping() const override { return "\01"; } | |
| 476 Char do_thousands_sep() const override { return ','; } | |
| 477 }; | |
| 478 | |
| 479 TEST(locale_test, localized_double) { | |
| 480 auto loc = std::locale(std::locale(), new numpunct<char>()); | |
| 481 EXPECT_EQ(fmt::format(loc, "{:L}", 1.23), "1?23"); | |
| 482 EXPECT_EQ(fmt::format(loc, "{:Lf}", 1.23), "1?230000"); | |
| 483 EXPECT_EQ(fmt::format(loc, "{:L}", 1234.5), "1~234?5"); | |
| 484 EXPECT_EQ(fmt::format(loc, "{:L}", 12000.0), "12~000"); | |
| 485 EXPECT_EQ(fmt::format(loc, "{:8L}", 1230.0), " 1~230"); | |
| 486 EXPECT_EQ(fmt::format(loc, "{:15.6Lf}", 0.1), " 0?100000"); | |
| 487 EXPECT_EQ(fmt::format(loc, "{:15.6Lf}", 1.0), " 1?000000"); | |
| 488 EXPECT_EQ(fmt::format(loc, "{:15.6Lf}", 1e3), " 1~000?000000"); | |
| 489 } | |
| 490 | |
| 491 TEST(locale_test, format) { | |
| 492 auto loc = std::locale(std::locale(), new numpunct<char>()); | |
| 493 EXPECT_EQ("1234567", fmt::format(std::locale(), "{:L}", 1234567)); | |
| 494 EXPECT_EQ("1~234~567", fmt::format(loc, "{:L}", 1234567)); | |
| 495 EXPECT_EQ("-1~234~567", fmt::format(loc, "{:L}", -1234567)); | |
| 496 EXPECT_EQ("-256", fmt::format(loc, "{:L}", -256)); | |
| 497 auto n = 1234567; | |
| 498 EXPECT_EQ("1~234~567", fmt::vformat(loc, "{:L}", fmt::make_format_args(n))); | |
| 499 auto s = std::string(); | |
| 500 fmt::format_to(std::back_inserter(s), loc, "{:L}", 1234567); | |
| 501 EXPECT_EQ("1~234~567", s); | |
| 502 | |
| 503 auto no_grouping_loc = std::locale(std::locale(), new no_grouping<char>()); | |
| 504 EXPECT_EQ("1234567", fmt::format(no_grouping_loc, "{:L}", 1234567)); | |
| 505 | |
| 506 auto special_grouping_loc = | |
| 507 std::locale(std::locale(), new special_grouping<char>()); | |
| 508 EXPECT_EQ("1,23,45,678", fmt::format(special_grouping_loc, "{:L}", 12345678)); | |
| 509 EXPECT_EQ("12,345", fmt::format(special_grouping_loc, "{:L}", 12345)); | |
| 510 | |
| 511 auto small_grouping_loc = | |
| 512 std::locale(std::locale(), new small_grouping<char>()); | |
| 513 EXPECT_EQ("4,2,9,4,9,6,7,2,9,5", | |
| 514 fmt::format(small_grouping_loc, "{:L}", max_value<uint32_t>())); | |
| 515 } | |
| 516 | |
| 517 TEST(locale_test, format_detault_align) { | |
| 518 auto loc = std::locale({}, new special_grouping<char>()); | |
| 519 EXPECT_EQ(" 12,345", fmt::format(loc, "{:8L}", 12345)); | |
| 520 } | |
| 521 | |
| 522 TEST(locale_test, format_plus) { | |
| 523 auto loc = std::locale({}, new special_grouping<char>()); | |
| 524 EXPECT_EQ("+100", fmt::format(loc, "{:+L}", 100)); | |
| 525 } | |
| 526 | |
| 527 TEST(locale_test, wformat) { | |
| 528 auto loc = std::locale(std::locale(), new numpunct<wchar_t>()); | |
| 529 EXPECT_EQ(L"1234567", fmt::format(std::locale(), L"{:L}", 1234567)); | |
| 530 EXPECT_EQ(L"1~234~567", fmt::format(loc, L"{:L}", 1234567)); | |
| 531 int n = 1234567; | |
| 532 EXPECT_EQ(L"1~234~567", | |
| 533 fmt::vformat(loc, L"{:L}", fmt::make_wformat_args(n))); | |
| 534 EXPECT_EQ(L"1234567", fmt::format(std::locale("C"), L"{:L}", 1234567)); | |
| 535 | |
| 536 auto no_grouping_loc = std::locale(std::locale(), new no_grouping<wchar_t>()); | |
| 537 EXPECT_EQ(L"1234567", fmt::format(no_grouping_loc, L"{:L}", 1234567)); | |
| 538 | |
| 539 auto special_grouping_loc = | |
| 540 std::locale(std::locale(), new special_grouping<wchar_t>()); | |
| 541 EXPECT_EQ(L"1,23,45,678", | |
| 542 fmt::format(special_grouping_loc, L"{:L}", 12345678)); | |
| 543 | |
| 544 auto small_grouping_loc = | |
| 545 std::locale(std::locale(), new small_grouping<wchar_t>()); | |
| 546 EXPECT_EQ(L"4,2,9,4,9,6,7,2,9,5", | |
| 547 fmt::format(small_grouping_loc, L"{:L}", max_value<uint32_t>())); | |
| 548 } | |
| 549 | |
| 550 TEST(locale_test, int_formatter) { | |
| 551 auto loc = std::locale(std::locale(), new special_grouping<char>()); | |
| 552 auto f = fmt::formatter<int>(); | |
| 553 auto parse_ctx = fmt::format_parse_context("L"); | |
| 554 f.parse(parse_ctx); | |
| 555 auto buf = fmt::memory_buffer(); | |
| 556 fmt::basic_format_context<fmt::appender, char> format_ctx( | |
| 557 fmt::appender(buf), {}, fmt::detail::locale_ref(loc)); | |
| 558 f.format(12345, format_ctx); | |
| 559 EXPECT_EQ(fmt::to_string(buf), "12,345"); | |
| 560 } | |
| 561 | |
| 562 FMT_BEGIN_NAMESPACE | |
| 563 template <class charT> struct formatter<std::complex<double>, charT> { | |
| 564 private: | |
| 565 detail::dynamic_format_specs<char> specs_; | |
| 566 | |
| 567 public: | |
| 568 FMT_CONSTEXPR typename basic_format_parse_context<charT>::iterator parse( | |
| 569 basic_format_parse_context<charT>& ctx) { | |
| 570 auto end = parse_format_specs(ctx.begin(), ctx.end(), specs_, ctx, | |
| 571 detail::type::float_type); | |
| 572 detail::parse_float_type_spec(specs_); | |
| 573 return end; | |
| 574 } | |
| 575 | |
| 576 template <class FormatContext> | |
| 577 typename FormatContext::iterator format(const std::complex<double>& c, | |
| 578 FormatContext& ctx) { | |
| 579 detail::handle_dynamic_spec<detail::precision_checker>( | |
| 580 specs_.precision, specs_.precision_ref, ctx); | |
| 581 auto specs = std::string(); | |
| 582 if (specs_.precision > 0) specs = fmt::format(".{}", specs_.precision); | |
| 583 if (specs_.type == presentation_type::fixed_lower) specs += 'f'; | |
| 584 auto real = fmt::format(ctx.locale().template get<std::locale>(), | |
| 585 fmt::runtime("{:" + specs + "}"), c.real()); | |
| 586 auto imag = fmt::format(ctx.locale().template get<std::locale>(), | |
| 587 fmt::runtime("{:" + specs + "}"), c.imag()); | |
| 588 auto fill_align_width = std::string(); | |
| 589 if (specs_.width > 0) fill_align_width = fmt::format(">{}", specs_.width); | |
| 590 return fmt::format_to(ctx.out(), runtime("{:" + fill_align_width + "}"), | |
| 591 c.real() != 0 ? fmt::format("({}+{}i)", real, imag) | |
| 592 : fmt::format("{}i", imag)); | |
| 593 } | |
| 594 }; | |
| 595 FMT_END_NAMESPACE | |
| 596 | |
| 597 TEST(locale_test, complex) { | |
| 598 std::string s = fmt::format("{}", std::complex<double>(1, 2)); | |
| 599 EXPECT_EQ(s, "(1+2i)"); | |
| 600 EXPECT_EQ(fmt::format("{:.2f}", std::complex<double>(1, 2)), "(1.00+2.00i)"); | |
| 601 EXPECT_EQ(fmt::format("{:8}", std::complex<double>(1, 2)), " (1+2i)"); | |
| 602 } | |
| 603 | |
| 604 TEST(locale_test, chrono_weekday) { | |
| 605 auto loc = get_locale("es_ES.UTF-8", "Spanish_Spain.1252"); | |
| 606 auto loc_old = std::locale::global(loc); | |
| 607 auto sat = fmt::weekday(6); | |
| 608 EXPECT_EQ(fmt::format(L"{}", sat), L"Sat"); | |
| 609 if (loc != std::locale::classic()) { | |
| 610 // L'\xE1' is 'á'. | |
| 611 auto saturdays = std::vector<std::wstring>{L"s\xE1""b", L"s\xE1."}; | |
| 612 EXPECT_THAT(saturdays, Contains(fmt::format(loc, L"{:L}", sat))); | |
| 613 } | |
| 614 std::locale::global(loc_old); | |
| 615 } | |
| 616 | |
| 617 TEST(locale_test, sign) { | |
| 618 EXPECT_EQ(fmt::format(std::locale(), L"{:L}", -50), L"-50"); | |
| 619 } | |
| 620 | |
| 621 TEST(std_test_xchar, optional) { | |
| 622 # ifdef __cpp_lib_optional | |
| 623 EXPECT_EQ(fmt::format(L"{}", std::optional{L'C'}), L"optional(\'C\')"); | |
| 624 EXPECT_EQ(fmt::format(L"{}", std::optional{std::wstring{L"wide string"}}), | |
| 625 L"optional(\"wide string\")"); | |
| 626 # endif | |
| 627 } | |
| 628 | |
| 629 #endif // FMT_STATIC_THOUSANDS_SEPARATOR | 
