mirror of
https://github.com/PurpleI2P/Boost-for-Android-Prebuilt
synced 2025-01-08 22:08:01 +00:00
182 lines
5.0 KiB
C++
182 lines
5.0 KiB
C++
/*=============================================================================
|
|
Copyright (c) 2015 Paul Fultz II
|
|
repeat_while.h
|
|
Distributed under 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)
|
|
==============================================================================*/
|
|
|
|
#ifndef BOOST_HOF_GUARD_REPEAT_WHILE_H
|
|
#define BOOST_HOF_GUARD_REPEAT_WHILE_H
|
|
|
|
/// repeat_while
|
|
/// ======
|
|
///
|
|
/// Description
|
|
/// -----------
|
|
///
|
|
/// The `repeat_while` function decorator will repeatedly apply a function while
|
|
/// the predicate returns a boolean that is true. If the predicate returns an
|
|
/// `IntergralConstant` then the predicate is only evaluated at compile-time.
|
|
///
|
|
///
|
|
/// Synopsis
|
|
/// --------
|
|
///
|
|
/// template<class Predicate>
|
|
/// constexpr auto repeat_while(Predicate predicate);
|
|
///
|
|
/// Requirements
|
|
/// ------------
|
|
///
|
|
/// Predicate must be:
|
|
///
|
|
/// * [ConstFunctionObject](ConstFunctionObject)
|
|
/// * MoveConstructible
|
|
///
|
|
/// Example
|
|
/// -------
|
|
///
|
|
/// #include <boost/hof.hpp>
|
|
/// #include <cassert>
|
|
///
|
|
/// struct increment
|
|
/// {
|
|
/// template<class T>
|
|
/// constexpr std::integral_constant<int, T::value + 1> operator()(T) const
|
|
/// {
|
|
/// return std::integral_constant<int, T::value + 1>();
|
|
/// }
|
|
/// };
|
|
///
|
|
/// struct not_6
|
|
/// {
|
|
/// template<class T>
|
|
/// constexpr std::integral_constant<bool, (T::value != 6)>
|
|
/// operator()(T) const
|
|
/// {
|
|
/// return std::integral_constant<bool, (T::value != 6)>();
|
|
/// }
|
|
/// };
|
|
///
|
|
/// typedef std::integral_constant<int, 1> one;
|
|
/// typedef std::integral_constant<int, 6> six;
|
|
///
|
|
/// int main() {
|
|
/// auto increment_until_6 = boost::hof::repeat_while(not_6())(increment());
|
|
/// static_assert(std::is_same<six, decltype(increment_until_6(one()))>::value, "Error");
|
|
/// }
|
|
///
|
|
|
|
#include <boost/hof/always.hpp>
|
|
#include <boost/hof/detail/delegate.hpp>
|
|
#include <boost/hof/detail/result_of.hpp>
|
|
#include <boost/hof/detail/move.hpp>
|
|
#include <boost/hof/decorate.hpp>
|
|
#include <boost/hof/detail/static_const_var.hpp>
|
|
#include <boost/hof/first_of.hpp>
|
|
#include <boost/hof/detail/recursive_constexpr_depth.hpp>
|
|
|
|
namespace boost { namespace hof { namespace detail {
|
|
|
|
template<class P, class... Ts>
|
|
struct compute_predicate
|
|
{
|
|
typedef decltype(std::declval<P>()(std::declval<Ts>()...)) type;
|
|
};
|
|
|
|
template<bool B>
|
|
struct while_repeater
|
|
{
|
|
template<class F, class P, class... Ts>
|
|
constexpr BOOST_HOF_SFINAE_RESULT(while_repeater<
|
|
compute_predicate<P, typename result_of<const F&, id_<Ts>...>::type>::type::value
|
|
>, id_<const F&>, id_<const P&>, result_of<const F&, id_<Ts>...>)
|
|
operator()(const F& f, const P& p, Ts&&... xs) const BOOST_HOF_SFINAE_RETURNS
|
|
(
|
|
while_repeater<
|
|
compute_predicate<P, decltype(f(BOOST_HOF_FORWARD(Ts)(xs)...))>::type::value
|
|
>()(f, p, f(BOOST_HOF_FORWARD(Ts)(xs)...))
|
|
);
|
|
};
|
|
|
|
template<>
|
|
struct while_repeater<false>
|
|
{
|
|
template<class F, class P, class T>
|
|
constexpr T operator()(const F&, const P&, T&& x) const
|
|
BOOST_HOF_RETURNS_DEDUCE_NOEXCEPT(x)
|
|
{
|
|
return x;
|
|
}
|
|
};
|
|
|
|
struct repeat_while_constant_decorator
|
|
{
|
|
template<class P, class F, class... Ts>
|
|
constexpr auto operator()(const P& p, const F& f, Ts&&... xs) const BOOST_HOF_RETURNS
|
|
(
|
|
detail::while_repeater<
|
|
detail::compute_predicate<P, decltype(std::declval<F>()(BOOST_HOF_FORWARD(Ts)(xs)...))>::type::value
|
|
>()
|
|
(
|
|
f,
|
|
p,
|
|
BOOST_HOF_FORWARD(Ts)(xs)...
|
|
)
|
|
);
|
|
};
|
|
|
|
template<int Depth>
|
|
struct repeat_while_integral_decorator
|
|
{
|
|
template<class P, class F, class T, class... Ts, class Self=repeat_while_integral_decorator<Depth-1>>
|
|
constexpr auto operator()(const P& p, const F& f, T&& x, Ts&&... xs) const BOOST_HOF_RETURNS
|
|
(
|
|
(p(x, BOOST_HOF_FORWARD(Ts)(xs)...)) ?
|
|
Self()(
|
|
p,
|
|
f,
|
|
f(x, BOOST_HOF_FORWARD(Ts)(xs)...)
|
|
) :
|
|
BOOST_HOF_FORWARD(T)(x)
|
|
);
|
|
};
|
|
|
|
template<>
|
|
struct repeat_while_integral_decorator<0>
|
|
{
|
|
template<class P, class F, class T, class Self=repeat_while_integral_decorator<0>>
|
|
#if BOOST_HOF_HAS_RELAXED_CONSTEXPR
|
|
constexpr
|
|
#endif
|
|
auto operator()(const P& p, const F& f, T x) const
|
|
BOOST_HOF_RETURNS_DEDUCE_NOEXCEPT((p(x), f(x)))
|
|
-> decltype(f(x))
|
|
{
|
|
while(p(x))
|
|
{
|
|
// TODO: Should move?
|
|
x = f(x);
|
|
}
|
|
return x;
|
|
}
|
|
};
|
|
}
|
|
|
|
#if BOOST_HOF_HAS_RELAXED_CONSTEXPR
|
|
#define BOOST_HOF_REPEAT_WHILE_CONSTEXPR_DEPTH 1
|
|
#else
|
|
#define BOOST_HOF_REPEAT_WHILE_CONSTEXPR_DEPTH BOOST_HOF_RECURSIVE_CONSTEXPR_DEPTH
|
|
#endif
|
|
|
|
BOOST_HOF_DECLARE_STATIC_VAR(repeat_while, decorate_adaptor<
|
|
boost::hof::first_of_adaptor<
|
|
detail::repeat_while_constant_decorator,
|
|
detail::repeat_while_integral_decorator<BOOST_HOF_REPEAT_WHILE_CONSTEXPR_DEPTH>
|
|
>
|
|
>);
|
|
|
|
}} // namespace boost::hof
|
|
|
|
#endif
|