123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123 |
- /*
- @Copyright Barrett Adair 2015-2017
- Distributed under the Boost Software License, Version 1.0.
- (See accompanying file LICENSE.md or copy at http://boost.org/LICENSE_1_0.txt)
- */
- #ifndef BOOST_CLBL_TRTS_APPLY_MEMBER_POINTER_HPP
- #define BOOST_CLBL_TRTS_APPLY_MEMBER_POINTER_HPP
- #include <boost/callable_traits/detail/core.hpp>
- namespace boost { namespace callable_traits {
- BOOST_CLBL_TRTS_DEFINE_SFINAE_ERROR_ORIGIN(apply_member_pointer)
- BOOST_CLBL_TRTS_SFINAE_MSG(apply_member_pointer, members_cannot_have_a_type_of_void)
- BOOST_CLBL_TRTS_SFINAE_MSG(apply_member_pointer, second_template_argument_must_be_a_class_or_struct)
- namespace detail {
- template<typename T, typename C, bool = std::is_class<C>::value>
- struct make_member_pointer;
- template<typename T, typename C>
- struct make_member_pointer<T, C, true> {
- using type = typename std::remove_reference<T>::type C::*;
- };
- template<typename C>
- struct make_member_pointer<void, C, true> {
- using type = invalid_type;
- };
- template<typename T, typename C>
- struct make_member_pointer<T, C, false> {
- using type = error_type<T>;
- };
- template<typename T, typename C>
- using make_member_pointer_t = typename make_member_pointer<T, C>::type;
- }
- //[ apply_member_pointer_hpp
- /*`
- [section:ref_apply_member_pointer apply_member_pointer]
- [heading Header]
- ``#include <boost/callable_traits/apply_member_pointer.hpp>``
- [heading Definition]
- */
- template<typename T, typename C>
- using apply_member_pointer_t = //see below
- //<-
- detail::sfinae_try<
- detail::fallback_if_invalid<
- typename detail::traits<T>::template apply_member_pointer<C>,
- typename detail::make_member_pointer<T, C>::type>,
- detail::fail_when_same<void, T, members_cannot_have_a_type_of_void>,
- detail::fail_if<!std::is_class<C>::value,
- second_template_argument_must_be_a_class_or_struct> >;
- namespace detail {
- template<typename T, typename C, typename = std::false_type>
- struct apply_member_pointer_impl {};
- template<typename T, typename C>
- struct apply_member_pointer_impl <T, C, typename std::is_same<
- apply_member_pointer_t<T, C>, detail::dummy>::type>
- {
- using type = apply_member_pointer_t<T, C>;
- };
- }
- //->
- template<typename T, typename C>
- struct apply_member_pointer : detail::apply_member_pointer_impl<T, C> {};
- //<-
- }} // namespace boost::callable_traits
- //->
- /*`
- [heading Constraints]
- * `T` may be any type except `void`
- * `C` must be a user-defined type
- [heading Behavior]
- * A substitution failure occurs if the constraints are violated.
- * When `T` is a function, function pointer (unqualified), or function reference, then the aliased type is a member function pointer of `C` with the same parameters and return type.
- * When `T` is a member function pointer (unqualified) of any type, the aliased type is a member function pointer of `C` with the same parameters and return type.
- * Otherwise, the aliased type is a member data pointer equivalent to `std::remove_reference_t<T> C::*`.
- [heading Input/Output Examples]
- [table
- [[`T`] [`apply_member_pointer_t<T, foo>`]]
- [[`int()`] [`int(foo::*)()`]]
- [[`int (&)()`] [`int(foo::*)()`]]
- [[`int (*)()`] [`int(foo::*)()`]]
- [[`int(bar::*)()`] [`int(foo::*)()`]]
- [[`int(bar::*)() &`] [`int(foo::*)() &`]]
- [[`int(bar::*)() &&`] [`int(foo::*)() &&`]]
- [[`int(bar::*)() const`] [`int(foo::*)() const`]]
- [[`int(bar::*)() transaction_safe`] [`int(foo::*)() transaction_safe`]]
- [[`int bar::*`] [`int foo::*`]]
- [[`int`] [`int foo::*`]]
- [[`int &`] [`int foo::*`]]
- [[`const int &`] [`const int foo::*`]]
- [[`int (*const)()`] [`int (*const foo::*)()`]]
- [[`void`] [(substitution failure)]]
- ]
- [heading Example Program]
- [import ../example/apply_member_pointer.cpp]
- [apply_member_pointer]
- [endsect]
- */
- //]
- #endif
|