forked from boostorg/mp11
Add mp_bind_q, mp_bind_front(_q), mp_bind_back(_q), mp_transform_q
This commit is contained in:
@@ -80,6 +80,7 @@ template<template<class...> class F, template<class...> class L1, class... T1, t
|
|||||||
} // namespace detail
|
} // namespace detail
|
||||||
|
|
||||||
template<template<class...> class F, class... L> using mp_transform = typename detail::mp_transform_impl<F, L...>::type;
|
template<template<class...> class F, class... L> using mp_transform = typename detail::mp_transform_impl<F, L...>::type;
|
||||||
|
template<class Q, class... L> using mp_transform_q = mp_transform<Q::template fn, L...>;
|
||||||
|
|
||||||
namespace detail
|
namespace detail
|
||||||
{
|
{
|
||||||
|
@@ -16,6 +16,7 @@ namespace boost
|
|||||||
namespace mp11
|
namespace mp11
|
||||||
{
|
{
|
||||||
|
|
||||||
|
// mp_arg
|
||||||
template<std::size_t I> struct mp_arg
|
template<std::size_t I> struct mp_arg
|
||||||
{
|
{
|
||||||
template<class... T> using fn = mp_at_c<mp_list<T...>, I>;
|
template<class... T> using fn = mp_at_c<mp_list<T...>, I>;
|
||||||
@@ -31,6 +32,7 @@ using _7 = mp_arg<6>;
|
|||||||
using _8 = mp_arg<7>;
|
using _8 = mp_arg<7>;
|
||||||
using _9 = mp_arg<8>;
|
using _9 = mp_arg<8>;
|
||||||
|
|
||||||
|
// mp_bind
|
||||||
template<template<class...> class F, class... T> struct mp_bind;
|
template<template<class...> class F, class... T> struct mp_bind;
|
||||||
|
|
||||||
namespace detail
|
namespace detail
|
||||||
@@ -58,6 +60,45 @@ template<template<class...> class F, class... T> struct mp_bind
|
|||||||
template<class... U> using fn = F<typename detail::eval_bound_arg<T, U...>::type...>;
|
template<class... U> using fn = F<typename detail::eval_bound_arg<T, U...>::type...>;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
template<class Q, class... T> using mp_bind_q = mp_bind<Q::template fn, T...>;
|
||||||
|
|
||||||
|
// mp_bind_front
|
||||||
|
template<template<class...> class F, class... T> struct mp_bind_front
|
||||||
|
{
|
||||||
|
#if defined( BOOST_MSVC ) && BOOST_WORKAROUND( BOOST_MSVC, <= 1910 && BOOST_MSVC >= 1900 )
|
||||||
|
#else
|
||||||
|
private:
|
||||||
|
#endif
|
||||||
|
|
||||||
|
template<class... U> struct _fn { using type = F<T..., U...>; };
|
||||||
|
|
||||||
|
public:
|
||||||
|
|
||||||
|
// the indirection through _fn works around the language inability
|
||||||
|
// to expand U... into a fixed parameter list of an alias template
|
||||||
|
|
||||||
|
template<class... U> using fn = typename _fn<U...>::type;
|
||||||
|
};
|
||||||
|
|
||||||
|
template<class Q, class... T> using mp_bind_front_q = mp_bind_front<Q::template fn, T...>;
|
||||||
|
|
||||||
|
// mp_bind_back
|
||||||
|
template<template<class...> class F, class... T> struct mp_bind_back
|
||||||
|
{
|
||||||
|
#if defined( BOOST_MSVC ) && BOOST_WORKAROUND( BOOST_MSVC, <= 1910 && BOOST_MSVC >= 1900 )
|
||||||
|
#else
|
||||||
|
private:
|
||||||
|
#endif
|
||||||
|
|
||||||
|
template<class... U> struct _fn { using type = F<U..., T...>; };
|
||||||
|
|
||||||
|
public:
|
||||||
|
|
||||||
|
template<class... U> using fn = typename _fn<U...>::type;
|
||||||
|
};
|
||||||
|
|
||||||
|
template<class Q, class... T> using mp_bind_back_q = mp_bind_back<Q::template fn, T...>;
|
||||||
|
|
||||||
} // namespace mp11
|
} // namespace mp11
|
||||||
} // namespace boost
|
} // namespace boost
|
||||||
|
|
||||||
|
@@ -9,6 +9,8 @@
|
|||||||
// http://www.boost.org/LICENSE_1_0.txt
|
// http://www.boost.org/LICENSE_1_0.txt
|
||||||
|
|
||||||
#include <boost/mp11/integral.hpp>
|
#include <boost/mp11/integral.hpp>
|
||||||
|
#include <boost/config.hpp>
|
||||||
|
#include <boost/detail/workaround.hpp>
|
||||||
|
|
||||||
namespace boost
|
namespace boost
|
||||||
{
|
{
|
||||||
@@ -106,14 +108,17 @@ template<template<class...> class F, class... T> using mp_defer = mp_if<mp_valid
|
|||||||
// mp_quote
|
// mp_quote
|
||||||
template<template<class...> class F, class... T> struct mp_quote
|
template<template<class...> class F, class... T> struct mp_quote
|
||||||
{
|
{
|
||||||
|
#if defined( BOOST_MSVC ) && BOOST_WORKAROUND( BOOST_MSVC, <= 1910 && BOOST_MSVC >= 1900 )
|
||||||
|
#else
|
||||||
private:
|
private:
|
||||||
|
#endif
|
||||||
|
|
||||||
template<class... U> struct _fn { using type = F<T..., U...>; };
|
template<class... U> struct _fn { using type = F<T..., U...>; };
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
// the indirection through _fn works around the language inability
|
// the indirection through _fn works around the language inability
|
||||||
// to expand T.. to expand into a fixed parameter list of an alias template
|
// to expand T... into a fixed parameter list of an alias template
|
||||||
|
|
||||||
template<class... U> using fn = typename _fn<U...>::type;
|
template<class... U> using fn = typename _fn<U...>::type;
|
||||||
};
|
};
|
||||||
|
@@ -106,3 +106,6 @@ run mp_map_update.cpp : : : $(REQ) ;
|
|||||||
|
|
||||||
# bind
|
# bind
|
||||||
run mp_bind.cpp : : : $(REQ) ;
|
run mp_bind.cpp : : : $(REQ) ;
|
||||||
|
run mp_bind_q.cpp : : : $(REQ) ;
|
||||||
|
run mp_bind_front.cpp : : : $(REQ) ;
|
||||||
|
run mp_bind_back.cpp : : : $(REQ) ;
|
||||||
|
63
test/mp_bind_back.cpp
Normal file
63
test/mp_bind_back.cpp
Normal file
@@ -0,0 +1,63 @@
|
|||||||
|
|
||||||
|
// Copyright 2017 Peter Dimov.
|
||||||
|
//
|
||||||
|
// 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
|
||||||
|
|
||||||
|
|
||||||
|
#include <boost/mp11/bind.hpp>
|
||||||
|
#include <boost/mp11/utility.hpp>
|
||||||
|
#include <boost/mp11/algorithm.hpp>
|
||||||
|
#include <boost/core/lightweight_test_trait.hpp>
|
||||||
|
#include <type_traits>
|
||||||
|
|
||||||
|
template<class...> struct L {};
|
||||||
|
template<class, class> struct P {};
|
||||||
|
|
||||||
|
template<class T, class U> using is_base_of_t = typename std::is_base_of<T, U>::type;
|
||||||
|
|
||||||
|
struct B1 {};
|
||||||
|
struct B2 {};
|
||||||
|
struct D: B1, B2 {};
|
||||||
|
struct NB {};
|
||||||
|
|
||||||
|
int main()
|
||||||
|
{
|
||||||
|
using namespace boost::mp11;
|
||||||
|
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_back<L, char[1], char[2]>::fn<int[1], int[2], int[3]>, L<int[1], int[2], int[3], char[1], char[2]>>));
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_back_q<mp_quote<L>, char[1], char[2]>::fn<int[1], int[2], int[3]>, L<int[1], int[2], int[3], char[1], char[2]>>));
|
||||||
|
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_back<P, char[1]>::fn<int[1]>, P<int[1], char[1]>>));
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_back_q<mp_quote<P>, char[1]>::fn<int[1]>, P<int[1], char[1]>>));
|
||||||
|
|
||||||
|
//
|
||||||
|
|
||||||
|
using L1 = L<B1, B2, NB>;
|
||||||
|
|
||||||
|
{
|
||||||
|
using L2 = mp_transform<mp_bind_back<is_base_of_t, D>::fn, L1>;
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<L2, L<std::true_type, std::true_type, std::false_type>>));
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
using L2 = mp_transform_q<mp_bind_back<is_base_of_t, D>, L1>;
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<L2, L<std::true_type, std::true_type, std::false_type>>));
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
using L2 = mp_transform<mp_bind_back_q<mp_quote<is_base_of_t>, D>::fn, L1>;
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<L2, L<std::true_type, std::true_type, std::false_type>>));
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
using L2 = mp_transform_q<mp_bind_back_q<mp_quote<is_base_of_t>, D>, L1>;
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<L2, L<std::true_type, std::true_type, std::false_type>>));
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
|
||||||
|
return boost::report_errors();
|
||||||
|
}
|
63
test/mp_bind_front.cpp
Normal file
63
test/mp_bind_front.cpp
Normal file
@@ -0,0 +1,63 @@
|
|||||||
|
|
||||||
|
// Copyright 2017 Peter Dimov.
|
||||||
|
//
|
||||||
|
// 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
|
||||||
|
|
||||||
|
|
||||||
|
#include <boost/mp11/bind.hpp>
|
||||||
|
#include <boost/mp11/utility.hpp>
|
||||||
|
#include <boost/mp11/algorithm.hpp>
|
||||||
|
#include <boost/core/lightweight_test_trait.hpp>
|
||||||
|
#include <type_traits>
|
||||||
|
|
||||||
|
template<class...> struct L {};
|
||||||
|
template<class, class> struct P {};
|
||||||
|
|
||||||
|
template<class T, class U> using is_base_of_t = typename std::is_base_of<T, U>::type;
|
||||||
|
|
||||||
|
struct B {};
|
||||||
|
struct D1: B {};
|
||||||
|
struct D2: B {};
|
||||||
|
struct ND {};
|
||||||
|
|
||||||
|
int main()
|
||||||
|
{
|
||||||
|
using namespace boost::mp11;
|
||||||
|
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_front<L, char[1], char[2]>::fn<int[1], int[2], int[3]>, L<char[1], char[2], int[1], int[2], int[3]>>));
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_front_q<mp_quote<L>, char[1], char[2]>::fn<int[1], int[2], int[3]>, L<char[1], char[2], int[1], int[2], int[3]>>));
|
||||||
|
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_front<P, char[1]>::fn<int[1]>, P<char[1], int[1]>>));
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_front_q<mp_quote<P>, char[1]>::fn<int[1]>, P<char[1], int[1]>>));
|
||||||
|
|
||||||
|
//
|
||||||
|
|
||||||
|
using L1 = L<D1, D2, ND>;
|
||||||
|
|
||||||
|
{
|
||||||
|
using L2 = mp_transform<mp_bind_front<is_base_of_t, B>::fn, L1>;
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<L2, L<std::true_type, std::true_type, std::false_type>>));
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
using L2 = mp_transform_q<mp_bind_front<is_base_of_t, B>, L1>;
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<L2, L<std::true_type, std::true_type, std::false_type>>));
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
using L2 = mp_transform<mp_bind_front_q<mp_quote<is_base_of_t>, B>::fn, L1>;
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<L2, L<std::true_type, std::true_type, std::false_type>>));
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
using L2 = mp_transform_q<mp_bind_front_q<mp_quote<is_base_of_t>, B>, L1>;
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<L2, L<std::true_type, std::true_type, std::false_type>>));
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
|
||||||
|
return boost::report_errors();
|
||||||
|
}
|
62
test/mp_bind_q.cpp
Normal file
62
test/mp_bind_q.cpp
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
|
||||||
|
// Copyright 2017 Peter Dimov.
|
||||||
|
//
|
||||||
|
// 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
|
||||||
|
|
||||||
|
|
||||||
|
#include <boost/mp11/bind.hpp>
|
||||||
|
#include <boost/mp11/utility.hpp>
|
||||||
|
#include <boost/core/lightweight_test_trait.hpp>
|
||||||
|
#include <type_traits>
|
||||||
|
|
||||||
|
struct X1 {};
|
||||||
|
struct X2 {};
|
||||||
|
struct X3 {};
|
||||||
|
struct X4 {};
|
||||||
|
struct X5 {};
|
||||||
|
struct X6 {};
|
||||||
|
struct X7 {};
|
||||||
|
struct X8 {};
|
||||||
|
struct X9 {};
|
||||||
|
|
||||||
|
template<class T> using add_pointer = typename std::add_pointer<T>::type;
|
||||||
|
|
||||||
|
int main()
|
||||||
|
{
|
||||||
|
using namespace boost::mp11;
|
||||||
|
|
||||||
|
using Q_addp = mp_quote<add_pointer>;
|
||||||
|
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_q<Q_addp, _1>::fn<X1>, X1*>));
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_q<Q_addp, _1>::fn<X1, X2, X3, X4, X5, X6, X7, X8, X9>, X1*>));
|
||||||
|
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_q<Q_addp, _2>::fn<X1, X2>, X2*>));
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_q<Q_addp, _2>::fn<X1, X2, X3, X4, X5, X6, X7, X8, X9>, X2*>));
|
||||||
|
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_q<Q_addp, _3>::fn<X1, X2, X3>, X3*>));
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_q<Q_addp, _3>::fn<X1, X2, X3, X4, X5, X6, X7, X8, X9>, X3*>));
|
||||||
|
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_q<Q_addp, _4>::fn<X1, X2, X3, X4>, X4*>));
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_q<Q_addp, _4>::fn<X1, X2, X3, X4, X5, X6, X7, X8, X9>, X4*>));
|
||||||
|
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_q<Q_addp, _5>::fn<X1, X2, X3, X4, X5>, X5*>));
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_q<Q_addp, _5>::fn<X1, X2, X3, X4, X5, X6, X7, X8, X9>, X5*>));
|
||||||
|
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_q<Q_addp, _6>::fn<X1, X2, X3, X4, X5, X6>, X6*>));
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_q<Q_addp, _6>::fn<X1, X2, X3, X4, X5, X6, X7, X8, X9>, X6*>));
|
||||||
|
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_q<Q_addp, _7>::fn<X1, X2, X3, X4, X5, X6, X7>, X7*>));
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_q<Q_addp, _7>::fn<X1, X2, X3, X4, X5, X6, X7, X8, X9>, X7*>));
|
||||||
|
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_q<Q_addp, _8>::fn<X1, X2, X3, X4, X5, X6, X7, X8>, X8*>));
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_q<Q_addp, _8>::fn<X1, X2, X3, X4, X5, X6, X7, X8, X9>, X8*>));
|
||||||
|
|
||||||
|
BOOST_TEST_TRAIT_TRUE((std::is_same<mp_bind_q<Q_addp, _9>::fn<X1, X2, X3, X4, X5, X6, X7, X8, X9>, X9*>));
|
||||||
|
|
||||||
|
//
|
||||||
|
|
||||||
|
return boost::report_errors();
|
||||||
|
}
|
Reference in New Issue
Block a user