1
0
mirror of https://github.com/PurpleI2P/Boost-for-Android-Prebuilt synced 2025-01-24 05:25:26 +00:00

182 lines
5.0 KiB
C++
Raw Normal View History

/*=============================================================================
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