2 //===----------------------------------------------------------------------===//
4 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
5 // See https://llvm.org/LICENSE.txt for license information.
6 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
8 //===----------------------------------------------------------------------===//
10 #ifndef _LIBCPP_EXPERIMENTAL_ITERATOR
11 #define _LIBCPP_EXPERIMENTAL_ITERATOR
15 namespace experimental {
16 inline namespace fundamentals_v2 {
18 template <class DelimT, class charT = char, class traits = char_traits<charT>>
19 class ostream_joiner {
21 typedef charT char_type;
22 typedef traits traits_type;
23 typedef basic_ostream<charT, traits> ostream_type;
24 typedef output_iterator_tag iterator_category;
25 typedef void value_type;
26 typedef void difference_type;
28 typedef void reference;
30 ostream_joiner(ostream_type& s, const DelimT& delimiter);
31 ostream_joiner(ostream_type& s, DelimT&& delimiter);
34 ostream_joiner& operator=(const T& value);
36 ostream_joiner& operator*() noexcept;
37 ostream_joiner& operator++() noexcept;
38 ostream_joiner& operator++(int) noexcept;
40 ostream_type* out_stream; // exposition only
41 DelimT delim; // exposition only
42 bool first_element; // exposition only
45 template <class charT, class traits, class DelimT>
46 ostream_joiner<decay_t<DelimT>, charT, traits>
47 make_ostream_joiner(basic_ostream<charT, traits>& os, DelimT&& delimiter);
49 } // inline namespace fundamentals_v2
50 } // namespace experimental
55 #if __cplusplus < 201103L && defined(_LIBCPP_USE_FROZEN_CXX03_HEADERS)
56 # include <__cxx03/experimental/iterator>
59 # include <__memory/addressof.h>
60 # include <__ostream/basic_ostream.h>
61 # include <__string/char_traits.h>
62 # include <__type_traits/decay.h>
63 # include <__utility/forward.h>
64 # include <__utility/move.h>
67 # if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
68 # pragma GCC system_header
72 # include <__undef_macros>
74 # if _LIBCPP_STD_VER >= 14
76 _LIBCPP_BEGIN_NAMESPACE_LFTS
78 template <class _Delim, class _CharT = char, class _Traits = char_traits<_CharT>>
79 class ostream_joiner {
81 typedef _CharT char_type;
82 typedef _Traits traits_type;
83 typedef basic_ostream<char_type, traits_type> ostream_type;
84 typedef output_iterator_tag iterator_category;
85 typedef void value_type;
86 typedef void difference_type;
88 typedef void reference;
90 _LIBCPP_HIDE_FROM_ABI ostream_joiner(ostream_type& __os, _Delim&& __d)
91 : __output_iter_(std::addressof(__os)), __delim_(std::move(__d)), __first_(true) {}
93 _LIBCPP_HIDE_FROM_ABI ostream_joiner(ostream_type& __os, const _Delim& __d)
94 : __output_iter_(std::addressof(__os)), __delim_(__d), __first_(true) {}
96 template <typename _Tp>
97 _LIBCPP_HIDE_FROM_ABI ostream_joiner& operator=(const _Tp& __v) {
99 *__output_iter_ << __delim_;
101 *__output_iter_ << __v;
105 _LIBCPP_HIDE_FROM_ABI ostream_joiner& operator*() _NOEXCEPT { return *this; }
106 _LIBCPP_HIDE_FROM_ABI ostream_joiner& operator++() _NOEXCEPT { return *this; }
107 _LIBCPP_HIDE_FROM_ABI ostream_joiner& operator++(int) _NOEXCEPT { return *this; }
110 ostream_type* __output_iter_;
115 template <class _CharT, class _Traits, class _Delim>
116 _LIBCPP_HIDE_FROM_ABI ostream_joiner<__decay_t<_Delim>, _CharT, _Traits>
117 make_ostream_joiner(basic_ostream<_CharT, _Traits>& __os, _Delim&& __d) {
118 return ostream_joiner<__decay_t<_Delim>, _CharT, _Traits>(__os, std::forward<_Delim>(__d));
121 _LIBCPP_END_NAMESPACE_LFTS
123 # endif // _LIBCPP_STD_VER >= 14
127 # if !defined(_LIBCPP_REMOVE_TRANSITIVE_INCLUDES) && _LIBCPP_STD_VER <= 20
130 # include <type_traits>
132 #endif // __cplusplus < 201103L && defined(_LIBCPP_USE_FROZEN_CXX03_HEADERS)
134 #endif // _LIBCPP_EXPERIMENTAL_ITERATOR