Document join and relax its compiler requirements

This commit is contained in:
Victor Zverovich 2019-05-29 10:27:05 -07:00
parent ad360a62b0
commit afc571aedc
3 changed files with 32 additions and 18 deletions

View File

@ -180,10 +180,6 @@ You can also write a formatter for a hierarchy of classes::
fmt::print("{}", a); // prints "B" fmt::print("{}", a); // prints "B"
} }
This section shows how to define a custom format function for a user-defined
type. The next section describes how to get ``fmt`` to use a conventional stream
output ``operator<<`` when one is defined for a user-defined type.
Output Iterator Support Output Iterator Support
----------------------- -----------------------
@ -214,6 +210,10 @@ Utilities
.. doxygenfunction:: fmt::to_string_view(basic_string_view<Char>) .. doxygenfunction:: fmt::to_string_view(basic_string_view<Char>)
.. doxygenfunction:: fmt::join(const Range&, string_view)
.. doxygenfunction:: fmt::join(It, It, string_view)
.. doxygenclass:: fmt::basic_memory_buffer .. doxygenclass:: fmt::basic_memory_buffer
:protected-members: :protected-members:
:members: :members:

View File

@ -162,13 +162,6 @@ FMT_END_NAMESPACE
# endif # endif
#endif #endif
#if FMT_HAS_GXX_CXX11 || FMT_HAS_FEATURE(cxx_trailing_return) || \
FMT_MSC_VER >= 1600
# define FMT_USE_TRAILING_RETURN 1
#else
# define FMT_USE_TRAILING_RETURN 0
#endif
#ifdef FMT_USE_INT128 #ifdef FMT_USE_INT128
// Do nothing. // Do nothing.
#elif defined(__SIZEOF_INT128__) #elif defined(__SIZEOF_INT128__)
@ -291,6 +284,12 @@ FMT_CONSTEXPR T* end(T (&array)[N]) FMT_NOEXCEPT {
return array + N; return array + N;
} }
// An implementation of iterator_t for pre-C++20 compilers such as gcc 4.
template <typename T>
struct iterator_t {
typedef decltype(internal::begin(internal::declval<const T&>())) type;
};
// For std::result_of in gcc 4.4. // For std::result_of in gcc 4.4.
template <typename Result> struct function { template <typename Result> struct function {
template <typename T> struct result { typedef Result type; }; template <typename T> struct result { typedef Result type; };
@ -3302,6 +3301,10 @@ struct formatter<arg_join<It, Char>, Char>
} }
}; };
/**
Returns an object that formats the iterator range `[begin, end)` with elements
separated by `sep`.
*/
template <typename It> template <typename It>
arg_join<It, char> join(It begin, It end, string_view sep) { arg_join<It, char> join(It begin, It end, string_view sep) {
return arg_join<It, char>(begin, end, sep); return arg_join<It, char>(begin, end, sep);
@ -3312,17 +3315,28 @@ arg_join<It, wchar_t> join(It begin, It end, wstring_view sep) {
return arg_join<It, wchar_t>(begin, end, sep); return arg_join<It, wchar_t>(begin, end, sep);
} }
// The following causes ICE in gcc 4.4. // gcc 4.4 on join: internal compiler error: in tsubst_copy, at cp/pt.c:10122
#if FMT_USE_TRAILING_RETURN && (!FMT_GCC_VERSION || FMT_GCC_VERSION >= 405) #if !FMT_GCC_VERSION || FMT_GCC_VERSION >= 405
/**
\rst
Returns an object that formats `range` with elements separated by `sep`.
**Example**::
std::vector<int> v = {1, 2, 3};
fmt::print("{}", fmt::join(v, ", "));
// Output: "1, 2, 3"
\endrst
*/
template <typename Range> template <typename Range>
auto join(const Range& range, string_view sep) arg_join<typename internal::iterator_t<Range>::type, char>
-> arg_join<decltype(internal::begin(range)), char> { join(const Range& range, string_view sep) {
return join(internal::begin(range), internal::end(range), sep); return join(internal::begin(range), internal::end(range), sep);
} }
template <typename Range> template <typename Range>
auto join(const Range& range, wstring_view sep) arg_join<typename internal::iterator_t<Range>::type, wchar_t>
-> arg_join<decltype(internal::begin(range)), wchar_t> { join(const Range& range, wstring_view sep) {
return join(internal::begin(range), internal::end(range), sep); return join(internal::begin(range), internal::end(range), sep);
} }
#endif #endif

View File

@ -1793,7 +1793,7 @@ TEST(FormatTest, JoinArg) {
EXPECT_EQ(format("{}, {}", v3[0], v3[1]), EXPECT_EQ(format("{}, {}", v3[0], v3[1]),
format("{}", join(v3, v3 + 2, ", "))); format("{}", join(v3, v3 + 2, ", ")));
#if FMT_USE_TRAILING_RETURN && (!FMT_GCC_VERSION || FMT_GCC_VERSION >= 405) #if !FMT_GCC_VERSION || FMT_GCC_VERSION >= 405
EXPECT_EQ("(1, 2, 3)", format("({})", join(v1, ", "))); EXPECT_EQ("(1, 2, 3)", format("({})", join(v1, ", ")));
EXPECT_EQ("(+01.20, +03.40)", format("({:+06.2f})", join(v2, ", "))); EXPECT_EQ("(+01.20, +03.40)", format("({:+06.2f})", join(v2, ", ")));
#endif #endif