123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304 |
- // Copyright Neil Groves 2010. Use, modification and
- // distribution is subject to the Boost Software License, Version
- // 1.0. (See accompanying file LICENSE_1_0.txt or copy at
- // http://www.boost.org/LICENSE_1_0.txt)
- //
- //
- // For more information, see http://www.boost.org/libs/range/
- //
- #ifndef BOOST_RANGE_COMBINE_HPP
- #define BOOST_RANGE_COMBINE_HPP
- #include <boost/iterator/zip_iterator.hpp>
- #include <boost/tuple/tuple.hpp>
- #include <boost/range/iterator.hpp>
- #include <boost/range/iterator_range.hpp>
- #include <boost/type_traits/is_void.hpp>
- #include <boost/type_traits/is_same.hpp>
- #include <boost/mpl/eval_if.hpp>
- #include <boost/mpl/int.hpp>
- #include <boost/mpl/plus.hpp>
- #include <boost/mpl/arithmetic.hpp>
- #include <boost/config.hpp>
- namespace boost
- {
- namespace range_detail
- {
- struct void_ { typedef void_ type; };
- }
- template<> struct range_iterator< ::boost::range_detail::void_ >
- {
- typedef ::boost::tuples::null_type type;
- };
- namespace range_detail
- {
- inline ::boost::tuples::null_type range_begin( ::boost::range_detail::void_& )
- { return ::boost::tuples::null_type(); }
- inline ::boost::tuples::null_type range_begin( const ::boost::range_detail::void_& )
- { return ::boost::tuples::null_type(); }
- inline ::boost::tuples::null_type range_end( ::boost::range_detail::void_& )
- { return ::boost::tuples::null_type(); }
- inline ::boost::tuples::null_type range_end( const ::boost::range_detail::void_& )
- { return ::boost::tuples::null_type(); }
- template< class T >
- struct tuple_iter
- {
- typedef BOOST_DEDUCED_TYPENAME ::boost::mpl::eval_if_c<
- ::boost::is_same<T, ::boost::range_detail::void_ >::value,
- ::boost::mpl::identity< ::boost::tuples::null_type >,
- ::boost::range_iterator<T>
- >::type type;
- };
- template< class Rng1, class Rng2 >
- struct tuple_range
- {
- typedef BOOST_DEDUCED_TYPENAME ::boost::mpl::eval_if_c<
- ::boost::is_same<Rng1, ::boost::range_detail::void_ >::value,
- ::boost::range_detail::void_,
- ::boost::mpl::identity<Rng1>
- >::type type;
- };
- template
- <
- class R1,
- class R2,
- class R3,
- class R4,
- class R5,
- class R6
- >
- struct generate_tuple
- {
- typedef ::boost::tuples::tuple<
- BOOST_DEDUCED_TYPENAME tuple_iter<R1>::type,
- BOOST_DEDUCED_TYPENAME tuple_iter<R2>::type,
- BOOST_DEDUCED_TYPENAME tuple_iter<R3>::type,
- BOOST_DEDUCED_TYPENAME tuple_iter<R4>::type,
- BOOST_DEDUCED_TYPENAME tuple_iter<R5>::type,
- BOOST_DEDUCED_TYPENAME tuple_iter<R6>::type
- > type;
- static type begin( R1& r1, R2& r2, R3& r3, R4& r4, R5& r5, R6& r6 )
- {
- return ::boost::tuples::make_tuple( ::boost::begin(r1),
- ::boost::begin(r2),
- ::boost::begin(r3),
- ::boost::begin(r4),
- ::boost::begin(r5),
- ::boost::begin(r6) );
- }
- static type end( R1& r1, R2& r2, R3& r3, R4& r4, R5& r5, R6& r6 )
- {
- return ::boost::tuples::make_tuple( ::boost::end(r1),
- ::boost::end(r2),
- ::boost::end(r3),
- ::boost::end(r4),
- ::boost::end(r5),
- ::boost::end(r6) );
- }
- };
- template
- <
- class R1,
- class R2 = void_,
- class R3 = void_,
- class R4 = void_,
- class R5 = void_,
- class R6 = void_
- >
- struct zip_rng
- : iterator_range<
- zip_iterator<
- BOOST_DEDUCED_TYPENAME generate_tuple<R1,R2,R3,R4,R5,R6>::type
- >
- >
- {
- private:
- typedef generate_tuple<R1,R2,R3,R4,R5,R6> generator_t;
- typedef BOOST_DEDUCED_TYPENAME generator_t::type tuple_t;
- typedef zip_iterator<tuple_t> zip_iter_t;
- typedef iterator_range<zip_iter_t> base_t;
- public:
- zip_rng( R1& r1, R2& r2, R3& r3, R4& r4, R5& r5, R6& r6 )
- : base_t( zip_iter_t( generator_t::begin(r1,r2,r3,r4,r5,r6) ),
- zip_iter_t( generator_t::end(r1,r2,r3,r4,r5,r6) ) )
- {
- BOOST_ASSERT(::boost::distance(r1) <= ::boost::distance(r2));
- BOOST_ASSERT(::boost::distance(r1) <= ::boost::distance(r3));
- BOOST_ASSERT(::boost::distance(r1) <= ::boost::distance(r4));
- BOOST_ASSERT(::boost::distance(r1) <= ::boost::distance(r5));
- BOOST_ASSERT(::boost::distance(r1) <= ::boost::distance(r6));
- }
- template< class Zip, class Rng >
- zip_rng( Zip& z, Rng& r )
- : base_t( zip_iter_t( generator_t::begin( z, r ) ),
- zip_iter_t( generator_t::end( z, r ) ) )
- {
- // @todo: tuple::begin( should be overloaded for this situation
- }
- struct tuple_length : ::boost::tuples::length<tuple_t>
- { };
- template< unsigned N >
- struct get
- {
- template< class Z, class R >
- static BOOST_DEDUCED_TYPENAME ::boost::tuples::element<N,tuple_t>::type begin( Z& z, R& )
- {
- return get<N>( z.begin().get_iterator_tuple() );
- }
- template< class Z, class R >
- static BOOST_DEDUCED_TYPENAME ::boost::tuples::element<N,tuple_t>::type end( Z& z, R& r )
- {
- return get<N>( z.end().get_iterator_tuple() );
- }
- };
- };
- template< class Rng1, class Rng2 >
- struct zip_range
- : iterator_range<
- zip_iterator<
- ::boost::tuples::tuple<
- BOOST_DEDUCED_TYPENAME ::boost::range_iterator<Rng1>::type,
- BOOST_DEDUCED_TYPENAME ::boost::range_iterator<Rng2>::type
- >
- >
- >
- {
- private:
- typedef zip_iterator<
- ::boost::tuples::tuple<
- BOOST_DEDUCED_TYPENAME ::boost::range_iterator<Rng1>::type,
- BOOST_DEDUCED_TYPENAME ::boost::range_iterator<Rng2>::type
- >
- > zip_iter_t;
- typedef iterator_range<zip_iter_t> base_t;
- public:
- zip_range( Rng1& r1, Rng2& r2 )
- : base_t( zip_iter_t( ::boost::tuples::make_tuple(::boost::begin(r1),
- ::boost::begin(r2)) ),
- zip_iter_t( ::boost::tuples::make_tuple(::boost::end(r1),
- ::boost::end(r2)) ) )
- {
- BOOST_ASSERT(::boost::distance(r1) <= ::boost::distance(r2));
- }
- };
- template< class Rng1, class Rng2, class Rng3 >
- struct zip_range3
- : iterator_range<
- zip_iterator<
- ::boost::tuples::tuple<
- BOOST_DEDUCED_TYPENAME ::boost::range_iterator<Rng1>::type,
- BOOST_DEDUCED_TYPENAME ::boost::range_iterator<Rng2>::type,
- BOOST_DEDUCED_TYPENAME ::boost::range_iterator<Rng3>::type
- >
- >
- >
- {
- private:
- typedef zip_iterator<
- ::boost::tuples::tuple<
- BOOST_DEDUCED_TYPENAME ::boost::range_iterator<Rng1>::type,
- BOOST_DEDUCED_TYPENAME ::boost::range_iterator<Rng2>::type,
- BOOST_DEDUCED_TYPENAME ::boost::range_iterator<Rng3>::type
- >
- > zip_iter_t;
- typedef iterator_range<zip_iter_t> base_t;
- public:
- zip_range3( Rng1& r1, Rng2& r2, Rng3& r3 )
- : base_t( zip_iter_t( ::boost::tuples::make_tuple(::boost::begin(r1),
- ::boost::begin(r2),
- ::boost::begin(r3)) ),
- zip_iter_t( ::boost::tuples::make_tuple(::boost::end(r1),
- ::boost::end(r2),
- ::boost::end(r3)) )
- )
- {
- BOOST_ASSERT(::boost::distance(r1) <= ::boost::distance(r2));
- BOOST_ASSERT(::boost::distance(r1) <= ::boost::distance(r3));
- }
- };
- struct combine_tag {};
- template< class Rng >
- inline zip_rng<Rng>
- operator&( combine_tag, Rng& r )
- {
- return zip_rng<Rng>(r);
- }
- template< class Rng >
- inline iterator_range<const Rng>
- operator&( combine_tag, const Rng& r )
- {
- return iterator_range<const Rng>(r);
- }
- template
- <
- class R1,
- class R2,
- class R3,
- class R4,
- class R5,
- class Rng
- >
- inline BOOST_DEDUCED_TYPENAME zip_rng<R1,R2,R3,R4,R5>::next
- operator&( const zip_rng<R1,R2,R3,R4,R5>& zip,
- Rng& r )
- {
- return zip_rng<R1,R2,R3,R4,R5>::next( zip, r );
- }
- } // namespace range_detail
- template< class Rng1, class Rng2 >
- inline ::boost::range_detail::zip_range<Rng1, Rng2> combine( Rng1& r1, Rng2& r2 )
- {
- return ::boost::range_detail::zip_range<Rng1, Rng2>(r1, r2);
- }
- template< class Rng1, class Rng2 >
- inline ::boost::range_detail::zip_range<const Rng1, Rng2> combine( const Rng1& r1, Rng2& r2 )
- {
- return ::boost::range_detail::zip_range<const Rng1, Rng2>(r1, r2);
- }
- template< class Rng1, class Rng2 >
- inline ::boost::range_detail::zip_range<Rng1, const Rng2> combine( Rng1& r1, const Rng2& r2 )
- {
- return ::boost::range_detail::zip_range<Rng1, const Rng2>(r1, r2);
- }
- template< class Rng1, class Rng2 >
- inline ::boost::range_detail::zip_range<const Rng1, const Rng2> combine( const Rng1& r1, const Rng2& r2 )
- {
- return ::boost::range_detail::zip_range<const Rng1, const Rng2>(r1, r2);
- }
- } // namespace boost
- #endif
|