1
0
mirror of https://github.com/PurpleI2P/Boost-for-Android-Prebuilt synced 2025-01-08 22:08:01 +00:00
Boost-for-Android-Prebuilt/boost-1_72_0/include/boost/hof/capture.hpp
r4sas 93de5720b8
add boost 1.72.0
Signed-off-by: r4sas <r4sas@i2pmail.org>
2020-02-29 22:43:48 +00:00

190 lines
5.3 KiB
C++

/*=============================================================================
Copyright (c) 2015 Paul Fultz II
capture.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_CAPTURE_H
#define BOOST_HOF_GUARD_CAPTURE_H
#include <boost/hof/detail/callable_base.hpp>
#include <boost/hof/detail/compressed_pair.hpp>
#include <boost/hof/reveal.hpp>
#include <boost/hof/pack.hpp>
#include <boost/hof/always.hpp>
#include <boost/hof/detail/move.hpp>
#include <boost/hof/detail/result_type.hpp>
/// capture
/// =======
///
/// Description
/// -----------
///
/// The `capture` function decorator is used to capture values in a function.
/// It provides more flexibility in capturing than the lambda capture list in
/// C++. It provides a way to do move and perfect capturing. The values
/// captured are prepended to the argument list of the function that will be
/// called.
///
/// Synopsis
/// --------
///
/// // Capture by decaying each value
/// template<class... Ts>
/// constexpr auto capture(Ts&&... xs);
///
/// // Capture lvalues by reference and rvalue reference by reference
/// template<class... Ts>
/// constexpr auto capture_forward(Ts&&... xs);
///
/// // Capture lvalues by reference and rvalues by value.
/// template<class... Ts>
/// constexpr auto capture_basic(Ts&&... xs);
///
/// Semantics
/// ---------
///
/// assert(capture(xs...)(f)(ys...) == f(xs..., ys...));
///
///
/// Example
/// -------
///
/// #include <boost/hof.hpp>
/// #include <cassert>
///
/// struct sum_f
/// {
/// template<class T, class U>
/// T operator()(T x, U y) const
/// {
/// return x+y;
/// }
/// };
///
/// int main() {
/// auto add_one = boost::hof::capture(1)(sum_f());
/// assert(add_one(2) == 3);
/// }
///
namespace boost { namespace hof {
namespace detail {
template<class F, class Pack>
struct capture_invoke : detail::compressed_pair<detail::callable_base<F>, Pack>, detail::function_result_type<F>
{
typedef capture_invoke fit_rewritable1_tag;
typedef detail::compressed_pair<detail::callable_base<F>, Pack> base;
BOOST_HOF_INHERIT_CONSTRUCTOR(capture_invoke, base)
template<class... Ts>
constexpr const detail::callable_base<F>& base_function(Ts&&... xs) const noexcept
{
return this->first(xs...);
}
template<class... Ts>
constexpr const Pack& get_pack(Ts&&...xs) const noexcept
{
return this->second(xs...);
}
template<class Failure, class... Ts>
struct unpack_capture_failure
{
template<class... Us>
struct apply
{
typedef typename Failure::template of<Us..., Ts...> type;
};
};
struct capture_failure
{
template<class Failure>
struct apply
{
template<class... Ts>
struct of
: Pack::template apply<unpack_capture_failure<Failure, Ts...>>::type
{};
};
};
struct failure
: failure_map<capture_failure, detail::callable_base<F>>
{};
BOOST_HOF_RETURNS_CLASS(capture_invoke);
template<class... Ts>
constexpr BOOST_HOF_SFINAE_RESULT
(
typename result_of<decltype(boost::hof::pack_join),
id_<const Pack&>,
result_of<decltype(boost::hof::pack_forward), id_<Ts>...>
>::type,
id_<detail::callable_base<F>&&>
)
operator()(Ts&&... xs) const BOOST_HOF_SFINAE_RETURNS
(
boost::hof::pack_join
(
BOOST_HOF_MANGLE_CAST(const Pack&)(BOOST_HOF_CONST_THIS->get_pack(xs...)),
boost::hof::pack_forward(BOOST_HOF_FORWARD(Ts)(xs)...)
)
(BOOST_HOF_RETURNS_C_CAST(detail::callable_base<F>&&)(BOOST_HOF_CONST_THIS->base_function(xs...)))
);
};
template<class Pack>
struct capture_pack : Pack
{
BOOST_HOF_INHERIT_CONSTRUCTOR(capture_pack, Pack);
BOOST_HOF_RETURNS_CLASS(capture_pack);
// TODO: Should use rvalue ref qualifier
template<class F>
constexpr auto operator()(F f) const BOOST_HOF_SFINAE_RETURNS
(
capture_invoke<F, Pack>(BOOST_HOF_RETURNS_STATIC_CAST(F&&)(f),
BOOST_HOF_RETURNS_C_CAST(Pack&&)(
BOOST_HOF_RETURNS_STATIC_CAST(const Pack&)(*boost::hof::always(BOOST_HOF_CONST_THIS)(f))
)
)
);
};
struct make_capture_pack_f
{
template<class Pack>
constexpr capture_pack<Pack> operator()(Pack p) const
BOOST_HOF_NOEXCEPT_CONSTRUCTIBLE(capture_pack<Pack>, Pack&&)
{
return capture_pack<Pack>(static_cast<Pack&&>(p));
}
};
template<class F>
struct capture_f
{
template<class... Ts>
constexpr auto operator()(Ts&&... xs) const BOOST_HOF_RETURNS
(
BOOST_HOF_RETURNS_CONSTRUCT(make_capture_pack_f)()(BOOST_HOF_RETURNS_CONSTRUCT(F)()(BOOST_HOF_FORWARD(Ts)(xs)...))
);
};
}
BOOST_HOF_DECLARE_STATIC_VAR(capture_basic, detail::capture_f<detail::pack_basic_f>);
BOOST_HOF_DECLARE_STATIC_VAR(capture_forward, detail::capture_f<detail::pack_forward_f>);
BOOST_HOF_DECLARE_STATIC_VAR(capture, detail::capture_f<detail::pack_f>);
}} // namespace boost::hof
#endif