pair.hpp 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  1. /*!
  2. @file
  3. Defines `boost::hana::pair`.
  4. @copyright Louis Dionne 2013-2016
  5. Distributed under the Boost Software License, Version 1.0.
  6. (See accompanying file LICENSE.md or copy at http://boost.org/LICENSE_1_0.txt)
  7. */
  8. #ifndef BOOST_HANA_PAIR_HPP
  9. #define BOOST_HANA_PAIR_HPP
  10. #include <boost/hana/fwd/pair.hpp>
  11. #include <boost/hana/basic_tuple.hpp>
  12. #include <boost/hana/config.hpp>
  13. #include <boost/hana/detail/decay.hpp>
  14. #include <boost/hana/detail/intrinsics.hpp>
  15. #include <boost/hana/detail/operators/adl.hpp>
  16. #include <boost/hana/detail/operators/comparable.hpp>
  17. #include <boost/hana/detail/operators/orderable.hpp>
  18. #include <boost/hana/fwd/core/make.hpp>
  19. #include <boost/hana/fwd/first.hpp>
  20. #include <boost/hana/fwd/second.hpp>
  21. #include <type_traits>
  22. #include <utility>
  23. BOOST_HANA_NAMESPACE_BEGIN
  24. //////////////////////////////////////////////////////////////////////////
  25. // pair
  26. //////////////////////////////////////////////////////////////////////////
  27. //! @cond
  28. template <typename First, typename Second>
  29. struct pair : detail::operators::adl<pair<First, Second>> {
  30. template <typename ...dummy, typename = typename std::enable_if<
  31. BOOST_HANA_TT_IS_CONSTRUCTIBLE(First, dummy...) &&
  32. BOOST_HANA_TT_IS_CONSTRUCTIBLE(Second, dummy...)
  33. >::type>
  34. constexpr pair()
  35. : storage_()
  36. { }
  37. template <typename ...dummy, typename = typename std::enable_if<
  38. BOOST_HANA_TT_IS_CONSTRUCTIBLE(First, First const&, dummy...) &&
  39. BOOST_HANA_TT_IS_CONSTRUCTIBLE(Second, Second const&, dummy...)
  40. >::type>
  41. constexpr pair(First const& first, Second const& second)
  42. : storage_{first, second}
  43. { }
  44. template <typename T, typename U, typename = typename std::enable_if<
  45. BOOST_HANA_TT_IS_CONVERTIBLE(T&&, First) &&
  46. BOOST_HANA_TT_IS_CONVERTIBLE(U&&, Second)
  47. >::type>
  48. constexpr pair(T&& t, U&& u)
  49. : storage_{static_cast<T&&>(t), static_cast<U&&>(u)}
  50. { }
  51. template <typename T, typename U, typename = typename std::enable_if<
  52. BOOST_HANA_TT_IS_CONVERTIBLE(T const&, First) &&
  53. BOOST_HANA_TT_IS_CONVERTIBLE(U const&, Second)
  54. >::type>
  55. constexpr pair(pair<T, U> const& other)
  56. : storage_{hana::get_impl<0>(other.storage_),
  57. hana::get_impl<1>(other.storage_)}
  58. { }
  59. template <typename T, typename U, typename = typename std::enable_if<
  60. BOOST_HANA_TT_IS_CONVERTIBLE(T&&, First) &&
  61. BOOST_HANA_TT_IS_CONVERTIBLE(U&&, Second)
  62. >::type>
  63. constexpr pair(pair<T, U>&& other)
  64. : storage_{static_cast<T&&>(hana::get_impl<0>(other.storage_)),
  65. static_cast<U&&>(hana::get_impl<1>(other.storage_))}
  66. { }
  67. template <typename T, typename U, typename = typename std::enable_if<
  68. BOOST_HANA_TT_IS_ASSIGNABLE(First&, T const&) &&
  69. BOOST_HANA_TT_IS_ASSIGNABLE(Second&, U const&)
  70. >::type>
  71. constexpr pair& operator=(pair<T, U> const& other) {
  72. hana::get_impl<0>(storage_) = hana::get_impl<0>(other.storage_);
  73. hana::get_impl<1>(storage_) = hana::get_impl<1>(other.storage_);
  74. return *this;
  75. }
  76. template <typename T, typename U, typename = typename std::enable_if<
  77. BOOST_HANA_TT_IS_ASSIGNABLE(First&, T&&) &&
  78. BOOST_HANA_TT_IS_ASSIGNABLE(Second&, U&&)
  79. >::type>
  80. constexpr pair& operator=(pair<T, U>&& other) {
  81. hana::get_impl<0>(storage_) = static_cast<T&&>(hana::get_impl<0>(other.storage_));
  82. hana::get_impl<1>(storage_) = static_cast<U&&>(hana::get_impl<1>(other.storage_));
  83. return *this;
  84. }
  85. using hana_tag = pair_tag;
  86. basic_tuple<First, Second> storage_;
  87. };
  88. //! @endcond
  89. //////////////////////////////////////////////////////////////////////////
  90. // Operators
  91. //////////////////////////////////////////////////////////////////////////
  92. namespace detail {
  93. template <>
  94. struct comparable_operators<pair_tag> {
  95. static constexpr bool value = true;
  96. };
  97. template <>
  98. struct orderable_operators<pair_tag> {
  99. static constexpr bool value = true;
  100. };
  101. }
  102. //////////////////////////////////////////////////////////////////////////
  103. // Product
  104. //////////////////////////////////////////////////////////////////////////
  105. template <>
  106. struct make_impl<pair_tag> {
  107. template <typename F, typename S>
  108. static constexpr pair<
  109. typename detail::decay<F>::type,
  110. typename detail::decay<S>::type
  111. > apply(F&& f, S&& s) {
  112. return {static_cast<F&&>(f), static_cast<S&&>(s)};
  113. }
  114. };
  115. template <>
  116. struct first_impl<pair_tag> {
  117. template <typename P>
  118. static constexpr decltype(auto) apply(P&& p)
  119. { return hana::get_impl<0>(static_cast<P&&>(p).storage_); }
  120. };
  121. template <>
  122. struct second_impl<pair_tag> {
  123. template <typename P>
  124. static constexpr decltype(auto) apply(P&& p)
  125. { return hana::get_impl<1>(static_cast<P&&>(p).storage_); }
  126. };
  127. BOOST_HANA_NAMESPACE_END
  128. #endif // !BOOST_HANA_PAIR_HPP