#ifndef SPROUT_ALGORITHM_STRING_JOIN_HPP #define SPROUT_ALGORITHM_STRING_JOIN_HPP #include #include #include #include #include #include #include #include #include #include #include #include namespace sprout { namespace algorithm { namespace detail { template struct string_size; template struct string_size< String, typename std::enable_if::value>::type > : public std::integral_constant< typename sprout::container_traits::size_type, sprout::container_traits::static_size - 1 > {}; template struct string_size< String, typename std::enable_if::value>::type > : public std::integral_constant< typename sprout::container_traits::size_type, sprout::container_traits::static_size > {}; template< typename String, typename sprout::enabler_if::value>::type = sprout::enabler > inline SPROUT_CONSTEXPR typename sprout::container_traits::difference_type str_size(String const& str) { return sprout::size(str) - 1; } template< typename String, typename sprout::enabler_if::value>::type = sprout::enabler > inline SPROUT_CONSTEXPR typename sprout::container_traits::difference_type str_size(String const& str) { return sprout::size(str); } } // namespace detail namespace result_of { // // join // template struct join { public: typedef typename sprout::container_transform_traits< typename sprout::container_traits::value_type >::template rebind_size< sprout::container_traits< typename sprout::container_traits::value_type >::static_size ? (sprout::container_traits< typename sprout::container_traits::value_type >::static_size + sprout::algorithm::detail::string_size::value ) * sprout::container_traits::static_size - sprout::algorithm::detail::string_size::value : 0 >::type type; }; template struct join { public: typedef typename sprout::container_transform_traits< typename sprout::container_traits::value_type >::template rebind_size< sprout::container_traits< typename sprout::container_traits::value_type >::static_size * sprout::container_traits::static_size >::type type; }; } // namespace result_of namespace detail { template inline SPROUT_CONSTEXPR typename sprout::container_traits::value_type join_impl_ra_2( ContIterator first_cont, SizeIterator found, Sizes const& sizes, sprout::index_t idx ) { typedef typename sprout::container_traits::value_type value_type; return found == sizes.end() ? value_type() : sprout::begin(first_cont[found - sizes.begin()])[idx - (found != sizes.begin() ? found[-1] : 0)] ; } template inline SPROUT_CONSTEXPR Result join_impl_ra_1( ContIterator first_cont, sprout::index_tuple, Sizes const& sizes ) { return sprout::make( sprout::algorithm::detail::join_impl_ra_2( first_cont, sprout::range::lower_bound(sizes, Indexes + 1), sizes, Indexes )... ); } template inline SPROUT_CONSTEXPR Result join_impl_ra(ContainerContainer const& cont_cont) { typedef typename sprout::container_traits::difference_type size_type; typedef sprout::array< size_type, sprout::container_traits::static_size > sizes_type; return sprout::algorithm::detail::join_impl_ra_1( sprout::begin(cont_cont), sprout::index_range<0, sprout::container_traits::static_size>::make(), sprout::range::partial_sum( cont_cont | sprout::adaptors::size_enumed, sprout::pit() ) ); } template inline SPROUT_CONSTEXPR typename std::enable_if< sprout::container_traits::static_size == sizeof...(Args), Result >::type join_impl( ContIterator first_cont, ContIterator last_cont, Args const&... args ); template inline SPROUT_CONSTEXPR typename std::enable_if< sprout::container_traits::static_size != sizeof...(Args), Result >::type join_impl( ContIterator first_cont, ContIterator last_cont, Args const&... args ); template inline SPROUT_CONSTEXPR typename std::enable_if< sprout::container_traits::static_size == sizeof...(Args), Result >::type join_impl_1( ContIterator first_cont, ContIterator last_cont, InputIterator first, InputIterator last, Args const&... args ) { return sprout::make(args...); } template inline SPROUT_CONSTEXPR typename std::enable_if< sprout::container_traits::static_size != sizeof...(Args), Result >::type join_impl_1( ContIterator first_cont, ContIterator last_cont, InputIterator first, InputIterator last, Args const&... args ) { return first != last ? sprout::algorithm::detail::join_impl_1( first_cont, last_cont, sprout::next(first), last, args..., *first ) : sprout::algorithm::detail::join_impl(sprout::next(first_cont), last_cont, args...) ; } template inline SPROUT_CONSTEXPR typename std::enable_if< sprout::container_traits::static_size == sizeof...(Args), Result >::type join_impl( ContIterator first_cont, ContIterator last_cont, Args const&... args ) { return sprout::make(args...); } template inline SPROUT_CONSTEXPR typename std::enable_if< sprout::container_traits::static_size != sizeof...(Args), Result >::type join_impl( ContIterator first_cont, ContIterator last_cont, Args const&... args ) { return first_cont != last_cont ? sprout::algorithm::detail::join_impl_1( first_cont, last_cont, sprout::begin(*first_cont), sprout::end(*first_cont), args... ) : sprout::make(args...) ; } template< typename ContainerContainer, typename sprout::enabler_if< sprout::is_random_access_iterator< typename sprout::container_traits::iterator >::value && sprout::is_random_access_iterator< typename sprout::container_traits< typename sprout::container_traits::value_type >::iterator >::value >::type = sprout::enabler > inline SPROUT_CONSTEXPR typename sprout::algorithm::result_of::join::type join(ContainerContainer const& cont_cont) { typedef typename sprout::algorithm::result_of::join::type result_type; return sprout::algorithm::detail::join_impl_ra(cont_cont); } template< typename ContainerContainer, typename sprout::enabler_if::iterator >::value && sprout::is_random_access_iterator< typename sprout::container_traits< typename sprout::container_traits::value_type >::iterator >::value )>::type = sprout::enabler > inline SPROUT_CONSTEXPR typename sprout::algorithm::result_of::join::type join(ContainerContainer const& cont_cont) { typedef typename sprout::algorithm::result_of::join::type result_type; return sprout::algorithm::detail::join_impl( sprout::begin(cont_cont), sprout::end(cont_cont) ); } } // namespace detail // // join // template inline SPROUT_CONSTEXPR typename sprout::algorithm::result_of::join::type join(ContainerContainer const& cont_cont) { return sprout::algorithm::detail::join(cont_cont); } namespace detail { template inline SPROUT_CONSTEXPR typename sprout::container_traits::value_type join_impl_ra_2( ContIterator first_cont, SepIterator first, SizeIterator found, Sizes const& sizes, sprout::index_t idx ) { typedef typename sprout::container_traits::value_type value_type; return found == sizes.end() ? value_type() : (found - sizes.begin()) % 2 ? first[idx - found[-1]] : sprout::begin(first_cont[(found - sizes.begin()) / 2])[idx - (found != sizes.begin() ? found[-1] : 0)] ; } template inline SPROUT_CONSTEXPR Result join_impl_ra_1( ContIterator first_cont, SepIterator first, sprout::index_tuple, Sizes const& sizes ) { return sprout::make( sprout::algorithm::detail::join_impl_ra_2( first_cont, first, sprout::range::lower_bound(sizes, Indexes + 1), sizes, Indexes )... ); } template inline SPROUT_CONSTEXPR Result join_impl_ra(ContainerContainer const& cont_cont, Separator const& separator) { typedef typename sprout::container_traits::difference_type size_type; typedef sprout::array< size_type, sprout::container_traits::static_size ? sprout::container_traits::static_size * 2 - 1 : 0 > sizes_type; return sprout::algorithm::detail::join_impl_ra_1( sprout::begin(cont_cont), sprout::begin(separator), sprout::index_range<0, sprout::container_traits::static_size>::make(), sprout::range::partial_sum( cont_cont | sprout::adaptors::size_enumed(sprout::algorithm::detail::str_size(separator), true), sprout::pit() ) ); } template inline SPROUT_CONSTEXPR typename std::enable_if< sprout::container_traits::static_size == sizeof...(Args), Result >::type join_impl( ContIterator first_cont, ContIterator last_cont, Args const&... args ); template inline SPROUT_CONSTEXPR typename std::enable_if< sprout::container_traits::static_size != sizeof...(Args), Result >::type join_impl( ContIterator first_cont, ContIterator last_cont, Args const&... args ); template inline SPROUT_CONSTEXPR typename std::enable_if< sprout::container_traits::static_size == sizeof...(Args), Result >::type join_impl_1( ContIterator first_cont, ContIterator last_cont, SepIterator sep_first, SepIterator sep_last, bool sep, InputIterator first, InputIterator last, Args const&... args ) { return sprout::make(args...); } template inline SPROUT_CONSTEXPR typename std::enable_if< sprout::container_traits::static_size != sizeof...(Args), Result >::type join_impl_1( ContIterator first_cont, ContIterator last_cont, SepIterator sep_first, SepIterator sep_last, bool sep, InputIterator first, InputIterator last, Args const&... args ) { return first != last ? sprout::algorithm::detail::join_impl_1( first_cont, last_cont, sep_first, sep_last, sep, sprout::next(first), last, args..., *first ) : sep ? sprout::algorithm::detail::join_impl( sprout::next(first_cont), last_cont, sep_first, sep_last, false, args... ) : sprout::algorithm::detail::join_impl( first_cont, last_cont, sep_first, sep_last, true, args... ) ; } template inline SPROUT_CONSTEXPR typename std::enable_if< sprout::container_traits::static_size == sizeof...(Args), Result >::type join_impl( ContIterator first_cont, ContIterator last_cont, SepIterator sep_first, SepIterator sep_last, bool sep, Args const&... args ) { return sprout::make(args...); } template inline SPROUT_CONSTEXPR typename std::enable_if< sprout::container_traits::static_size != sizeof...(Args), Result >::type join_impl( ContIterator first_cont, ContIterator last_cont, SepIterator sep_first, SepIterator sep_last, bool sep, Args const&... args ) { return first_cont != last_cont ? sep ? sprout::algorithm::detail::join_impl_1( first_cont, last_cont, sep_first, sep_last, sep, sep_first, sep_last, args... ) : sprout::algorithm::detail::join_impl_1( first_cont, last_cont, sep_first, sep_last, sep, sprout::begin(*first_cont), sprout::end(*first_cont), args... ) : sprout::make(args...) ; } template< typename ContainerContainer, typename Separator, typename sprout::enabler_if< sprout::is_random_access_iterator< typename sprout::container_traits::iterator >::value && sprout::is_random_access_iterator< typename sprout::container_traits< typename sprout::container_traits::value_type >::iterator >::value && sprout::is_random_access_iterator< typename sprout::container_traits::iterator >::value >::type = sprout::enabler > inline SPROUT_CONSTEXPR typename sprout::algorithm::result_of::join::type join(ContainerContainer const& cont_cont, Separator const& separator) { typedef typename sprout::algorithm::result_of::join::type result_type; return sprout::algorithm::detail::join_impl_ra(cont_cont, separator); } template< typename ContainerContainer, typename Separator, typename sprout::enabler_if::iterator >::value && sprout::is_random_access_iterator< typename sprout::container_traits< typename sprout::container_traits::value_type >::iterator >::value && sprout::is_random_access_iterator< typename sprout::container_traits::iterator >::value )>::type = sprout::enabler > inline SPROUT_CONSTEXPR typename sprout::algorithm::result_of::join::type join(ContainerContainer const& cont_cont, Separator const& separator) { typedef typename sprout::algorithm::result_of::join::type result_type; return sprout::algorithm::detail::join_impl( sprout::begin(cont_cont), sprout::end(cont_cont), sprout::begin(separator), sprout::end(separator), false ); } } // namespace detail // // join // template inline SPROUT_CONSTEXPR typename sprout::algorithm::result_of::join::type join(ContainerContainer const& cont_cont, Separator const& separator) { return sprout::algorithm::detail::join(cont_cont, separator); } } // namespace algorithm } // namespace sprout #endif // #ifndef SPROUT_ALGORITHM_STRING_JOIN_HPP