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 |