1
0
Fork 0
mirror of https://github.com/bolero-MURAKAMI/Sprout synced 2024-11-12 21:09:01 +00:00
Sprout/sprout/iterator/const_reference_cast.hpp
2016-04-19 15:14:40 +09:00

242 lines
8.4 KiB
C++

/*=============================================================================
Copyright (c) 2011-2016 Bolero MURAKAMI
https://github.com/bolero-MURAKAMI/Sprout
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 SPROUT_ITERATOR_CONST_REFERENCE_CAST_HPP
#define SPROUT_ITERATOR_CONST_REFERENCE_CAST_HPP
#include <type_traits>
#include <sprout/config.hpp>
#include <sprout/type_traits/integral_constant.hpp>
#include <sprout/type_traits/is_const_cast_convertible.hpp>
#include <sprout/type_traits/remove_cvref.hpp>
#include <sprout/type_traits/enabler_if.hpp>
#include <sprout/utility/forward.hpp>
#include <sprout/adl/not_found.hpp>
namespace sprout {
//
// is_const_reference_cast_convertible
//
template<typename From, typename To>
struct is_const_reference_cast_convertible
: public sprout::bool_constant<
sprout::is_const_cast_convertible<From, To>::value
|| std::is_same<typename sprout::remove_cvref<From>::type, typename std::remove_cv<To>::type>::value
>
{};
template<typename From, typename To>
struct is_const_reference_cast_convertible<From, To const>
: public sprout::is_const_reference_cast_convertible<From, To>
{};
template<typename From, typename To>
struct is_const_reference_cast_convertible<From, To volatile>
: public sprout::is_const_reference_cast_convertible<From, To>
{};
template<typename From, typename To>
struct is_const_reference_cast_convertible<From, To const volatile>
: public sprout::is_const_reference_cast_convertible<From, To>
{};
template<typename From, typename To>
struct is_const_reference_cast_convertible<From const, To>
: public sprout::is_const_reference_cast_convertible<From, To>
{};
template<typename From, typename To>
struct is_const_reference_cast_convertible<From const, To const>
: public sprout::is_const_reference_cast_convertible<From, To>
{};
template<typename From, typename To>
struct is_const_reference_cast_convertible<From const, To volatile>
: public sprout::is_const_reference_cast_convertible<From, To>
{};
template<typename From, typename To>
struct is_const_reference_cast_convertible<From const, To const volatile>
: public sprout::is_const_reference_cast_convertible<From, To>
{};
template<typename From, typename To>
struct is_const_reference_cast_convertible<From volatile, To>
: public sprout::is_const_reference_cast_convertible<From, To>
{};
template<typename From, typename To>
struct is_const_reference_cast_convertible<From volatile, To const>
: public sprout::is_const_reference_cast_convertible<From, To>
{};
template<typename From, typename To>
struct is_const_reference_cast_convertible<From volatile, To volatile>
: public sprout::is_const_reference_cast_convertible<From, To>
{};
template<typename From, typename To>
struct is_const_reference_cast_convertible<From volatile, To const volatile>
: public sprout::is_const_reference_cast_convertible<From, To>
{};
template<typename From, typename To>
struct is_const_reference_cast_convertible<From const volatile, To>
: public sprout::is_const_reference_cast_convertible<From, To>
{};
template<typename From, typename To>
struct is_const_reference_cast_convertible<From const volatile, To const>
: public sprout::is_const_reference_cast_convertible<From, To>
{};
template<typename From, typename To>
struct is_const_reference_cast_convertible<From const volatile, To volatile>
: public sprout::is_const_reference_cast_convertible<From, To>
{};
template<typename From, typename To>
struct is_const_reference_cast_convertible<From const volatile, To const volatile>
: public sprout::is_const_reference_cast_convertible<From, To>
{};
} // namespace sprout
namespace sprout_adl {
template<typename To>
sprout::not_found_via_adl const_reference_conversion(...);
} // namespace sprout_adl
namespace sprout {
namespace iterator_detail {
template<
typename T, typename U,
typename sprout::enabler_if<
sprout::is_const_reference_cast_convertible<U&, T>::value
&& std::is_reference<T>::value
>::type = sprout::enabler
>
inline SPROUT_CONSTEXPR T
const_reference_conversion(U& t) {
return const_cast<T>(t);
}
template<
typename T, typename U,
typename sprout::enabler_if<
sprout::is_const_reference_cast_convertible<U const&, T>::value
&& std::is_reference<T>::value
>::type = sprout::enabler
>
inline SPROUT_CONSTEXPR T
const_reference_conversion(U const& t) {
return const_cast<T>(t);
}
template<
typename T, typename U,
typename sprout::enabler_if<
sprout::is_const_reference_cast_convertible<U&, T>::value
&& std::is_pointer<T>::value
>::type = sprout::enabler
>
inline SPROUT_CONSTEXPR T
const_reference_conversion(U& t) {
return const_cast<T>(t);
}
template<
typename T, typename U,
typename sprout::enabler_if<
sprout::is_const_reference_cast_convertible<U const&, T>::value
&& std::is_pointer<T>::value
>::type = sprout::enabler
>
inline SPROUT_CONSTEXPR T
const_reference_conversion(U const& t) {
return const_cast<T>(t);
}
template<
typename T, typename U,
typename sprout::enabler_if<
sprout::is_const_reference_cast_convertible<U&, T>::value
&& !std::is_reference<T>::value && !std::is_pointer<T>::value
>::type = sprout::enabler
>
inline SPROUT_CONSTEXPR T
const_reference_conversion(U& t) {
return static_cast<T>(t);
}
template<
typename T, typename U,
typename sprout::enabler_if<
sprout::is_const_reference_cast_convertible<U const&, T>::value
&& !std::is_reference<T>::value && !std::is_pointer<T>::value
>::type = sprout::enabler
>
inline SPROUT_CONSTEXPR T
const_reference_conversion(U const& t) {
return static_cast<T>(t);
}
} // namespace iterator_detail
} // namespace sprout
namespace sprout_iterator_detail {
template<typename To, typename From>
inline SPROUT_CONSTEXPR To
call_const_reference_conversion(From&& t) {
using sprout::iterator_detail::const_reference_conversion;
using sprout_adl::const_reference_conversion;
return const_reference_conversion<To>(SPROUT_FORWARD(From, t));
}
} // namespace sprout_iterator_detail
namespace sprout {
//
// const_reference_cast
//
template<typename To, typename From>
inline SPROUT_CONSTEXPR To
const_reference_cast(From&& t) {
return sprout_iterator_detail::call_const_reference_conversion<To>(SPROUT_FORWARD(From, t));
}
} // namespace sprout
//
// note:
// const_reference_cast is an adaptable function for interconversion
// with reference and const_reference.
// If you want to adapt a user-defined reference proxy class to const_reference_cast:
// - Specialize sprout::is_const_reference_cast_convertible.
// - Overload const_reference_conversion as to be lookup in the ADL.
//
// example:
// #include <type_traits>
// #include <sprout/config.hpp>
// #include <sprout/type_traits.hpp>
// #include <sprout/iterator/const_reference_cast.hpp>
// /* Mylib::Reference is an user-defined reference proxy class*/
// namespace Mylib {
// template<typename T>
// struct Reference {
// T* p;
// typedef T* pointer;
// /* definition reference proxy interface... */
// };
// }
// /* const_reference_cast adapt for Mylib::Reference */
// namespace sprout {
// template<typename From, typename To>
// struct is_const_reference_cast_convertible<
// Mylib::Reference<From>,
// Mylib::Reference<To>
// >
// : public sprout::is_same<
// typename std::remove_cv<From>::type,
// typename std::remove_cv<To>::type
// >
// {};
// }
// namespace Mylib {
// template<typename To, typename From>
// inline SPROUT_CONSTEXPR typename std::enable_if<
// sprout::is_const_reference_cast_convertible<Reference<From>, To>::value,
// To
// >::type
// const_reference_conversion(Reference<From> const& it) {
// return To{const_cast<typename To::pointer>(it.p)};
// }
// }
// /* test const_reference_cast with Mylib::Reference */
// constexpr Mylib::Reference<int const> it{0};
// static_assert(sprout::const_reference_cast<Mylib::Reference<int> >(it).p == 0, "");
//
#endif // #ifndef SPROUT_ITERATOR_CONST_REFERENCE_CAST_HPP