keyed_element.hpp 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  1. /*=============================================================================
  2. Copyright (c) 2005-2012 Joel de Guzman
  3. Copyright (c) 2005-2006 Dan Marsden
  4. Distributed under the Boost Software License, Version 1.0. (See accompanying
  5. file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  6. ==============================================================================*/
  7. #if !defined(BOOST_FUSION_DEQUE_DETAIL_KEYED_ELEMENT_26112006_1330)
  8. #define BOOST_FUSION_DEQUE_DETAIL_KEYED_ELEMENT_26112006_1330
  9. #include <boost/fusion/support/config.hpp>
  10. #include <boost/fusion/support/detail/access.hpp>
  11. #include <boost/fusion/iterator/deref.hpp>
  12. #include <boost/fusion/iterator/next.hpp>
  13. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES) && BOOST_WORKAROUND(BOOST_GCC, / 100 == 404)
  14. #include <boost/core/enable_if.hpp>
  15. #include <boost/type_traits/is_same.hpp>
  16. #endif
  17. namespace boost { namespace fusion
  18. {
  19. struct fusion_sequence_tag;
  20. }}
  21. namespace boost { namespace fusion { namespace detail
  22. {
  23. struct nil_keyed_element
  24. {
  25. typedef fusion_sequence_tag tag;
  26. BOOST_FUSION_GPU_ENABLED
  27. void get();
  28. template<typename It>
  29. BOOST_CONSTEXPR BOOST_FUSION_GPU_ENABLED
  30. static nil_keyed_element
  31. from_iterator(It const&)
  32. {
  33. return nil_keyed_element();
  34. }
  35. };
  36. template <typename Key, typename Value, typename Rest>
  37. struct keyed_element : Rest
  38. {
  39. typedef Rest base;
  40. typedef fusion_sequence_tag tag;
  41. using Rest::get;
  42. template <typename It>
  43. BOOST_CONSTEXPR BOOST_FUSION_GPU_ENABLED
  44. static keyed_element
  45. from_iterator(It const& it)
  46. {
  47. return keyed_element(
  48. *it, base::from_iterator(fusion::next(it)));
  49. }
  50. BOOST_CONSTEXPR BOOST_FUSION_GPU_ENABLED
  51. keyed_element(keyed_element const& rhs)
  52. : Rest(rhs.get_base()), value_(rhs.value_)
  53. {}
  54. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  55. BOOST_CXX14_CONSTEXPR BOOST_FUSION_GPU_ENABLED
  56. keyed_element(keyed_element&& rhs)
  57. : Rest(rhs.forward_base())
  58. , value_(BOOST_FUSION_FWD_ELEM(Value, rhs.value_))
  59. {}
  60. #endif
  61. template <typename U, typename Rst>
  62. BOOST_CONSTEXPR BOOST_FUSION_GPU_ENABLED
  63. keyed_element(keyed_element<Key, U, Rst> const& rhs
  64. , typename enable_if<is_convertible<U, Value> >::type* = 0)
  65. : Rest(rhs.get_base()), value_(rhs.value_)
  66. {}
  67. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  68. #endif
  69. BOOST_CXX14_CONSTEXPR BOOST_FUSION_GPU_ENABLED
  70. Rest& get_base() BOOST_NOEXCEPT
  71. {
  72. return *this;
  73. }
  74. BOOST_CONSTEXPR BOOST_FUSION_GPU_ENABLED
  75. Rest const& get_base() const BOOST_NOEXCEPT
  76. {
  77. return *this;
  78. }
  79. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  80. BOOST_CXX14_CONSTEXPR BOOST_FUSION_GPU_ENABLED
  81. Rest&& forward_base() BOOST_NOEXCEPT
  82. {
  83. return std::move(*static_cast<Rest*>(this));
  84. }
  85. #endif
  86. BOOST_CONSTEXPR BOOST_FUSION_GPU_ENABLED
  87. typename cref_result<Value>::type get(Key) const
  88. {
  89. return value_;
  90. }
  91. BOOST_CXX14_CONSTEXPR BOOST_FUSION_GPU_ENABLED
  92. typename ref_result<Value>::type get(Key)
  93. {
  94. return value_;
  95. }
  96. BOOST_CONSTEXPR BOOST_FUSION_GPU_ENABLED
  97. keyed_element(
  98. typename detail::call_param<Value>::type value
  99. , Rest const& rest)
  100. : Rest(rest), value_(value)
  101. {}
  102. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  103. #if BOOST_WORKAROUND(BOOST_GCC, / 100 == 404)
  104. template <typename Value_, typename = typename enable_if<is_same<Value_, Value> >::type>
  105. #else
  106. typedef Value Value_;
  107. #endif
  108. BOOST_CXX14_CONSTEXPR BOOST_FUSION_GPU_ENABLED
  109. keyed_element(Value_&& value, Rest&& rest)
  110. : Rest(std::move(rest))
  111. , value_(BOOST_FUSION_FWD_ELEM(Value, value))
  112. {}
  113. #endif
  114. BOOST_CONSTEXPR BOOST_FUSION_GPU_ENABLED
  115. keyed_element()
  116. : Rest(), value_()
  117. {}
  118. template<typename U, typename Rst>
  119. BOOST_CXX14_CONSTEXPR BOOST_FUSION_GPU_ENABLED
  120. keyed_element& operator=(keyed_element<Key, U, Rst> const& rhs)
  121. {
  122. base::operator=(static_cast<Rst const&>(rhs)); // cast for msvc-7.1
  123. value_ = rhs.value_;
  124. return *this;
  125. }
  126. BOOST_CXX14_CONSTEXPR BOOST_FUSION_GPU_ENABLED
  127. keyed_element& operator=(keyed_element const& rhs)
  128. {
  129. base::operator=(rhs);
  130. value_ = rhs.value_;
  131. return *this;
  132. }
  133. #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
  134. BOOST_CXX14_CONSTEXPR BOOST_FUSION_GPU_ENABLED
  135. keyed_element& operator=(keyed_element&& rhs)
  136. {
  137. base::operator=(rhs.forward_base());
  138. value_ = std::move(rhs.value_);
  139. return *this;
  140. }
  141. #endif
  142. Value value_;
  143. };
  144. template<typename Elem, typename Key>
  145. struct keyed_element_value_at
  146. : keyed_element_value_at<typename Elem::base, Key>
  147. {};
  148. template<typename Key, typename Value, typename Rest>
  149. struct keyed_element_value_at<keyed_element<Key, Value, Rest>, Key>
  150. {
  151. typedef Value type;
  152. };
  153. }}}
  154. #endif