forked from boostorg/range
Fixed range adaptors that were creating underlying iterators that were not default constructible.
This commit is contained in:
@@ -12,6 +12,7 @@
|
||||
#define BOOST_RANGE_ADAPTOR_FILTERED_HPP
|
||||
|
||||
#include <boost/range/adaptor/argument_fwd.hpp>
|
||||
#include <boost/range/detail/default_constructible_unary_fn.hpp>
|
||||
#include <boost/range/iterator_range.hpp>
|
||||
#include <boost/range/concepts.hpp>
|
||||
#include <boost/iterator/filter_iterator.hpp>
|
||||
@@ -23,21 +24,28 @@ namespace boost
|
||||
template< class P, class R >
|
||||
struct filtered_range :
|
||||
boost::iterator_range<
|
||||
boost::filter_iterator< P,
|
||||
BOOST_DEDUCED_TYPENAME range_iterator<R>::type
|
||||
boost::filter_iterator<
|
||||
typename default_constructible_unary_fn_gen<P, bool>::type,
|
||||
typename range_iterator<R>::type
|
||||
>
|
||||
>
|
||||
{
|
||||
private:
|
||||
typedef boost::iterator_range<
|
||||
boost::filter_iterator< P,
|
||||
BOOST_DEDUCED_TYPENAME range_iterator<R>::type
|
||||
>
|
||||
> base;
|
||||
boost::filter_iterator<
|
||||
typename default_constructible_unary_fn_gen<P, bool>::type,
|
||||
typename range_iterator<R>::type
|
||||
>
|
||||
> base;
|
||||
public:
|
||||
filtered_range( P p, R& r )
|
||||
: base( make_filter_iterator( p, boost::begin(r), boost::end(r) ),
|
||||
make_filter_iterator( p, boost::end(r), boost::end(r) ) )
|
||||
typedef typename default_constructible_unary_fn_gen<P, bool>::type
|
||||
pred_t;
|
||||
|
||||
filtered_range(P p, R& r)
|
||||
: base(make_filter_iterator(pred_t(p),
|
||||
boost::begin(r), boost::end(r)),
|
||||
make_filter_iterator(pred_t(p),
|
||||
boost::end(r), boost::end(r)))
|
||||
{ }
|
||||
};
|
||||
|
||||
|
@@ -20,6 +20,7 @@
|
||||
#include <boost/range/concepts.hpp>
|
||||
#include <boost/iterator/iterator_adaptor.hpp>
|
||||
#include <boost/iterator/transform_iterator.hpp>
|
||||
#include <boost/optional/optional.hpp>
|
||||
|
||||
namespace boost
|
||||
{
|
||||
@@ -32,19 +33,36 @@ namespace boost
|
||||
typedef const Value& result_type;
|
||||
typedef const Value& first_argument_type;
|
||||
|
||||
// Rationale:
|
||||
// The default constructor is required to allow the transform
|
||||
// iterator to properly model the iterator concept.
|
||||
replace_value()
|
||||
{
|
||||
}
|
||||
|
||||
replace_value(const Value& from, const Value& to)
|
||||
: m_from(from), m_to(to)
|
||||
: m_impl(data(from, to))
|
||||
{
|
||||
}
|
||||
|
||||
const Value& operator()(const Value& x) const
|
||||
{
|
||||
return (x == m_from) ? m_to : x;
|
||||
return (x == m_impl->m_from) ? m_impl->m_to : x;
|
||||
}
|
||||
|
||||
private:
|
||||
Value m_from;
|
||||
Value m_to;
|
||||
struct data
|
||||
{
|
||||
data(const Value& from, const Value& to)
|
||||
: m_from(from)
|
||||
, m_to(to)
|
||||
{
|
||||
}
|
||||
|
||||
Value m_from;
|
||||
Value m_to;
|
||||
};
|
||||
boost::optional<data> m_impl;
|
||||
};
|
||||
|
||||
template< class R >
|
||||
|
@@ -20,6 +20,7 @@
|
||||
#include <boost/range/concepts.hpp>
|
||||
#include <boost/iterator/iterator_adaptor.hpp>
|
||||
#include <boost/iterator/transform_iterator.hpp>
|
||||
#include <boost/optional/optional.hpp>
|
||||
|
||||
namespace boost
|
||||
{
|
||||
@@ -32,19 +33,34 @@ namespace boost
|
||||
typedef const Value& result_type;
|
||||
typedef const Value& first_argument_type;
|
||||
|
||||
// Rationale:
|
||||
// required to allow the iterator to be default constructible.
|
||||
replace_value_if()
|
||||
{
|
||||
}
|
||||
|
||||
replace_value_if(const Pred& pred, const Value& to)
|
||||
: m_pred(pred), m_to(to)
|
||||
: m_impl(data(pred, to))
|
||||
{
|
||||
}
|
||||
|
||||
const Value& operator()(const Value& x) const
|
||||
{
|
||||
return m_pred(x) ? m_to : x;
|
||||
return m_impl->m_pred(x) ? m_impl->m_to : x;
|
||||
}
|
||||
|
||||
private:
|
||||
Pred m_pred;
|
||||
Value m_to;
|
||||
struct data
|
||||
{
|
||||
data(const Pred& p, const Value& t)
|
||||
: m_pred(p), m_to(t)
|
||||
{
|
||||
}
|
||||
|
||||
Pred m_pred;
|
||||
Value m_to;
|
||||
};
|
||||
boost::optional<data> m_impl;
|
||||
};
|
||||
|
||||
template< class Pred, class R >
|
||||
|
@@ -12,6 +12,7 @@
|
||||
#define BOOST_RANGE_ADAPTOR_TRANSFORMED_HPP
|
||||
|
||||
#include <boost/range/adaptor/argument_fwd.hpp>
|
||||
#include <boost/range/detail/default_constructible_unary_fn.hpp>
|
||||
#include <boost/range/iterator_range.hpp>
|
||||
#include <boost/range/concepts.hpp>
|
||||
#include <boost/iterator/transform_iterator.hpp>
|
||||
@@ -21,31 +22,46 @@ namespace boost
|
||||
{
|
||||
namespace range_detail
|
||||
{
|
||||
// A type generator to produce the transform_iterator type conditionally
|
||||
// including a wrapped predicate as appropriate.
|
||||
template<typename P, typename It>
|
||||
struct transform_iterator_gen
|
||||
{
|
||||
typedef transform_iterator<
|
||||
typename default_constructible_unary_fn_gen<
|
||||
P,
|
||||
typename transform_iterator<P, It>::reference
|
||||
>::type,
|
||||
It
|
||||
> type;
|
||||
};
|
||||
|
||||
template< class F, class R >
|
||||
struct transformed_range :
|
||||
public boost::iterator_range<
|
||||
boost::transform_iterator< F,
|
||||
BOOST_DEDUCED_TYPENAME range_iterator<R>::type
|
||||
>
|
||||
>
|
||||
typename transform_iterator_gen<
|
||||
F, typename range_iterator<R>::type>::type>
|
||||
{
|
||||
private:
|
||||
typedef boost::iterator_range<
|
||||
boost::transform_iterator< F,
|
||||
BOOST_DEDUCED_TYPENAME range_iterator<R>::type
|
||||
>
|
||||
>
|
||||
base;
|
||||
typedef typename transform_iterator_gen<
|
||||
F, typename range_iterator<R>::type>::type transform_iter_t;
|
||||
|
||||
typedef boost::iterator_range<transform_iter_t> base;
|
||||
|
||||
public:
|
||||
typedef F transform_fn_type;
|
||||
typedef typename default_constructible_unary_fn_gen<
|
||||
F,
|
||||
typename transform_iterator<
|
||||
F,
|
||||
typename range_iterator<R>::type
|
||||
>::reference
|
||||
>::type transform_fn_type;
|
||||
|
||||
typedef R source_range_type;
|
||||
|
||||
transformed_range( F f, R& r )
|
||||
: base( boost::make_transform_iterator( boost::begin(r), f ),
|
||||
boost::make_transform_iterator( boost::end(r), f ) )
|
||||
|
||||
transformed_range(transform_fn_type f, R& r)
|
||||
: base(transform_iter_t(boost::begin(r), f),
|
||||
transform_iter_t(boost::end(r), f))
|
||||
{
|
||||
}
|
||||
};
|
||||
|
@@ -0,0 +1,64 @@
|
||||
// Boost.Range library
|
||||
//
|
||||
// Copyright Neil Groves 2014. 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_DETAIL_DEFAULT_CONSTRUCTIBLE_UNARY_FN_HPP_INCLUDED
|
||||
#define BOOST_RANGE_DETAIL_DEFAULT_CONSTRUCTIBLE_UNARY_FN_HPP_INCLUDED
|
||||
|
||||
#include <boost/optional/optional.hpp>
|
||||
#include <boost/mpl/if.hpp>
|
||||
#include <boost/type_traits/has_trivial_constructor.hpp>
|
||||
|
||||
namespace boost
|
||||
{
|
||||
namespace range_detail
|
||||
{
|
||||
|
||||
template<typename F, typename R>
|
||||
class default_constructible_unary_fn_wrapper
|
||||
{
|
||||
public:
|
||||
typedef R result_type;
|
||||
|
||||
default_constructible_unary_fn_wrapper()
|
||||
{
|
||||
}
|
||||
default_constructible_unary_fn_wrapper(const F& source)
|
||||
: m_impl(source)
|
||||
{
|
||||
}
|
||||
template<typename Arg>
|
||||
R operator()(const Arg& arg) const
|
||||
{
|
||||
BOOST_ASSERT(m_impl);
|
||||
return (*m_impl)(arg);
|
||||
}
|
||||
template<typename Arg>
|
||||
R operator()(Arg& arg) const
|
||||
{
|
||||
BOOST_ASSERT(m_impl);
|
||||
return (*m_impl)(arg);
|
||||
}
|
||||
private:
|
||||
boost::optional<F> m_impl;
|
||||
};
|
||||
|
||||
template<typename F, typename R>
|
||||
struct default_constructible_unary_fn_gen
|
||||
{
|
||||
typedef typename boost::mpl::if_<
|
||||
boost::has_trivial_default_constructor<F>,
|
||||
F,
|
||||
default_constructible_unary_fn_wrapper<F,R>
|
||||
>::type type;
|
||||
};
|
||||
|
||||
} // namespace range_detail
|
||||
} // namespace boost
|
||||
|
||||
#endif // include guard
|
Reference in New Issue
Block a user