optional.hpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416
  1. /*!
  2. @file
  3. Defines `boost::hana::optional`.
  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_OPTIONAL_HPP
  9. #define BOOST_HANA_OPTIONAL_HPP
  10. #include <boost/hana/fwd/optional.hpp>
  11. #include <boost/hana/bool.hpp>
  12. #include <boost/hana/config.hpp>
  13. #include <boost/hana/core/tag_of.hpp>
  14. #include <boost/hana/detail/operators/adl.hpp>
  15. #include <boost/hana/detail/operators/comparable.hpp>
  16. #include <boost/hana/detail/operators/monad.hpp>
  17. #include <boost/hana/detail/operators/orderable.hpp>
  18. #include <boost/hana/detail/wrong.hpp>
  19. #include <boost/hana/functional/partial.hpp>
  20. #include <boost/hana/fwd/any_of.hpp>
  21. #include <boost/hana/fwd/ap.hpp>
  22. #include <boost/hana/fwd/concat.hpp>
  23. #include <boost/hana/fwd/core/make.hpp>
  24. #include <boost/hana/fwd/empty.hpp>
  25. #include <boost/hana/fwd/equal.hpp>
  26. #include <boost/hana/fwd/find_if.hpp>
  27. #include <boost/hana/fwd/flatten.hpp>
  28. #include <boost/hana/fwd/less.hpp>
  29. #include <boost/hana/fwd/lift.hpp>
  30. #include <boost/hana/fwd/transform.hpp>
  31. #include <boost/hana/fwd/type.hpp>
  32. #include <boost/hana/fwd/unpack.hpp>
  33. #include <cstddef> // std::nullptr_t
  34. #include <type_traits>
  35. #include <utility>
  36. BOOST_HANA_NAMESPACE_BEGIN
  37. //////////////////////////////////////////////////////////////////////////
  38. // optional<>
  39. //////////////////////////////////////////////////////////////////////////
  40. namespace detail {
  41. template <typename T, typename = typename hana::tag_of<T>::type>
  42. struct nested_type { };
  43. template <typename T>
  44. struct nested_type<T, type_tag> { using type = typename T::type; };
  45. }
  46. template <typename T>
  47. struct optional<T> : detail::operators::adl<>, detail::nested_type<T> {
  48. // 5.3.1, Constructors
  49. constexpr optional() = default;
  50. constexpr optional(optional const&) = default;
  51. constexpr optional(optional&&) = default;
  52. constexpr optional(T const& t)
  53. : value_(t)
  54. { }
  55. constexpr optional(T&& t)
  56. : value_(static_cast<T&&>(t))
  57. { }
  58. // 5.3.3, Assignment
  59. constexpr optional& operator=(optional const&) = default;
  60. constexpr optional& operator=(optional&&) = default;
  61. // 5.3.5, Observers
  62. constexpr T const* operator->() const { return &value_; }
  63. constexpr T* operator->() { return &value_; }
  64. constexpr T& value() & { return value_; }
  65. constexpr T const& value() const& { return value_; }
  66. constexpr T&& value() && { return static_cast<T&&>(value_); }
  67. constexpr T const&& value() const&& { return static_cast<T const&&>(value_); }
  68. constexpr T& operator*() & { return value_; }
  69. constexpr T const& operator*() const& { return value_; }
  70. constexpr T&& operator*() && { return static_cast<T&&>(value_); }
  71. constexpr T const&& operator*() const&& { return static_cast<T const&&>(value_); }
  72. template <typename U> constexpr T& value_or(U&&) & { return value_; }
  73. template <typename U> constexpr T const& value_or(U&&) const& { return value_; }
  74. template <typename U> constexpr T&& value_or(U&&) && { return static_cast<T&&>(value_); }
  75. template <typename U> constexpr T const&& value_or(U&&) const&& { return static_cast<T const&&>(value_); }
  76. // We leave this public because it simplifies the implementation, but
  77. // this should be considered private by users.
  78. T value_;
  79. };
  80. //! @cond
  81. template <typename ...dummy>
  82. constexpr auto optional<>::value() const {
  83. static_assert(detail::wrong<dummy...>{},
  84. "hana::optional::value() requires a non-empty optional");
  85. }
  86. template <typename ...dummy>
  87. constexpr auto optional<>::operator*() const {
  88. static_assert(detail::wrong<dummy...>{},
  89. "hana::optional::operator* requires a non-empty optional");
  90. }
  91. template <typename U>
  92. constexpr U&& optional<>::value_or(U&& u) const {
  93. return static_cast<U&&>(u);
  94. }
  95. template <typename T>
  96. constexpr auto make_just_t::operator()(T&& t) const {
  97. return hana::optional<typename std::decay<T>::type>(static_cast<T&&>(t));
  98. }
  99. //! @endcond
  100. template <typename ...T>
  101. struct tag_of<optional<T...>> {
  102. using type = optional_tag;
  103. };
  104. //////////////////////////////////////////////////////////////////////////
  105. // make<optional_tag>
  106. //////////////////////////////////////////////////////////////////////////
  107. template <>
  108. struct make_impl<optional_tag> {
  109. template <typename X>
  110. static constexpr auto apply(X&& x)
  111. { return hana::just(static_cast<X&&>(x)); }
  112. static constexpr auto apply()
  113. { return hana::nothing; }
  114. };
  115. //////////////////////////////////////////////////////////////////////////
  116. // Operators
  117. //////////////////////////////////////////////////////////////////////////
  118. namespace detail {
  119. template <>
  120. struct comparable_operators<optional_tag> {
  121. static constexpr bool value = true;
  122. };
  123. template <>
  124. struct orderable_operators<optional_tag> {
  125. static constexpr bool value = true;
  126. };
  127. template <>
  128. struct monad_operators<optional_tag> {
  129. static constexpr bool value = true;
  130. };
  131. }
  132. //////////////////////////////////////////////////////////////////////////
  133. // is_just and is_nothing
  134. //////////////////////////////////////////////////////////////////////////
  135. //! @cond
  136. template <typename ...T>
  137. constexpr auto is_just_t::operator()(optional<T...> const&) const
  138. { return hana::bool_c<sizeof...(T) != 0>; }
  139. template <typename ...T>
  140. constexpr auto is_nothing_t::operator()(optional<T...> const&) const
  141. { return hana::bool_c<sizeof...(T) == 0>; }
  142. //! @endcond
  143. //////////////////////////////////////////////////////////////////////////
  144. // sfinae
  145. //////////////////////////////////////////////////////////////////////////
  146. namespace detail {
  147. struct sfinae_impl {
  148. template <typename F, typename ...X, typename = decltype(
  149. std::declval<F>()(std::declval<X>()...)
  150. )>
  151. constexpr decltype(auto) operator()(int, F&& f, X&& ...x) const {
  152. using Return = decltype(static_cast<F&&>(f)(static_cast<X&&>(x)...));
  153. static_assert(!std::is_same<Return, void>::value,
  154. "hana::sfinae(f)(args...) requires f(args...) to be non-void");
  155. return hana::just(static_cast<F&&>(f)(static_cast<X&&>(x)...));
  156. }
  157. template <typename F, typename ...X>
  158. constexpr auto operator()(long, F&&, X&& ...) const
  159. { return hana::nothing; }
  160. };
  161. }
  162. //! @cond
  163. template <typename F>
  164. constexpr decltype(auto) sfinae_t::operator()(F&& f) const {
  165. return hana::partial(detail::sfinae_impl{}, int{},
  166. static_cast<F&&>(f));
  167. }
  168. //! @endcond
  169. //////////////////////////////////////////////////////////////////////////
  170. // Comparable
  171. //////////////////////////////////////////////////////////////////////////
  172. template <>
  173. struct equal_impl<optional_tag, optional_tag> {
  174. template <typename T, typename U>
  175. static constexpr auto apply(hana::optional<T> const& t, hana::optional<U> const& u)
  176. { return hana::equal(t.value_, u.value_); }
  177. static constexpr hana::true_ apply(hana::optional<> const&, hana::optional<> const&)
  178. { return {}; }
  179. template <typename T, typename U>
  180. static constexpr hana::false_ apply(T const&, U const&)
  181. { return {}; }
  182. };
  183. //////////////////////////////////////////////////////////////////////////
  184. // Orderable
  185. //////////////////////////////////////////////////////////////////////////
  186. template <>
  187. struct less_impl<optional_tag, optional_tag> {
  188. template <typename T>
  189. static constexpr hana::true_ apply(hana::optional<> const&, hana::optional<T> const&)
  190. { return {}; }
  191. static constexpr hana::false_ apply(hana::optional<> const&, hana::optional<> const&)
  192. { return {}; }
  193. template <typename T>
  194. static constexpr hana::false_ apply(hana::optional<T> const&, hana::optional<> const&)
  195. { return {}; }
  196. template <typename T, typename U>
  197. static constexpr auto apply(hana::optional<T> const& x, hana::optional<U> const& y)
  198. { return hana::less(x.value_, y.value_); }
  199. };
  200. //////////////////////////////////////////////////////////////////////////
  201. // Functor
  202. //////////////////////////////////////////////////////////////////////////
  203. template <>
  204. struct transform_impl<optional_tag> {
  205. template <typename F>
  206. static constexpr auto apply(optional<> const&, F&&)
  207. { return hana::nothing; }
  208. template <typename T, typename F>
  209. static constexpr auto apply(optional<T> const& opt, F&& f)
  210. { return hana::just(static_cast<F&&>(f)(opt.value_)); }
  211. template <typename T, typename F>
  212. static constexpr auto apply(optional<T>& opt, F&& f)
  213. { return hana::just(static_cast<F&&>(f)(opt.value_)); }
  214. template <typename T, typename F>
  215. static constexpr auto apply(optional<T>&& opt, F&& f)
  216. { return hana::just(static_cast<F&&>(f)(static_cast<T&&>(opt.value_))); }
  217. };
  218. //////////////////////////////////////////////////////////////////////////
  219. // Applicative
  220. //////////////////////////////////////////////////////////////////////////
  221. template <>
  222. struct lift_impl<optional_tag> {
  223. template <typename X>
  224. static constexpr auto apply(X&& x)
  225. { return hana::just(static_cast<X&&>(x)); }
  226. };
  227. template <>
  228. struct ap_impl<optional_tag> {
  229. template <typename F, typename X>
  230. static constexpr auto ap_helper(F&&, X&&, ...)
  231. { return hana::nothing; }
  232. template <typename F, typename X>
  233. static constexpr auto ap_helper(F&& f, X&& x, hana::true_, hana::true_)
  234. { return hana::just(static_cast<F&&>(f).value_(static_cast<X&&>(x).value_)); }
  235. template <typename F, typename X>
  236. static constexpr auto apply(F&& f, X&& x) {
  237. return ap_impl::ap_helper(static_cast<F&&>(f), static_cast<X&&>(x),
  238. hana::is_just(f), hana::is_just(x));
  239. }
  240. };
  241. //////////////////////////////////////////////////////////////////////////
  242. // Monad
  243. //////////////////////////////////////////////////////////////////////////
  244. template <>
  245. struct flatten_impl<optional_tag> {
  246. static constexpr auto apply(optional<> const&)
  247. { return hana::nothing; }
  248. static constexpr auto apply(optional<optional<>> const&)
  249. { return hana::nothing; }
  250. template <typename T>
  251. static constexpr auto apply(optional<optional<T>> const& opt)
  252. { return hana::just(opt.value_.value_); }
  253. template <typename T>
  254. static constexpr auto apply(optional<optional<T>>&& opt)
  255. { return hana::just(static_cast<T&&>(opt.value_.value_)); }
  256. };
  257. //////////////////////////////////////////////////////////////////////////
  258. // MonadPlus
  259. //////////////////////////////////////////////////////////////////////////
  260. template <>
  261. struct concat_impl<optional_tag> {
  262. template <typename Y>
  263. static constexpr auto apply(hana::optional<>&, Y&& y)
  264. { return static_cast<Y&&>(y); }
  265. template <typename Y>
  266. static constexpr auto apply(hana::optional<>&&, Y&& y)
  267. { return static_cast<Y&&>(y); }
  268. template <typename Y>
  269. static constexpr auto apply(hana::optional<> const&, Y&& y)
  270. { return static_cast<Y&&>(y); }
  271. template <typename X, typename Y>
  272. static constexpr auto apply(X&& x, Y&&)
  273. { return static_cast<X&&>(x); }
  274. };
  275. template <>
  276. struct empty_impl<optional_tag> {
  277. static constexpr auto apply()
  278. { return hana::nothing; }
  279. };
  280. //////////////////////////////////////////////////////////////////////////
  281. // Foldable
  282. //////////////////////////////////////////////////////////////////////////
  283. template <>
  284. struct unpack_impl<optional_tag> {
  285. template <typename T, typename F>
  286. static constexpr decltype(auto) apply(optional<T>&& opt, F&& f)
  287. { return static_cast<F&&>(f)(static_cast<T&&>(opt.value_)); }
  288. template <typename T, typename F>
  289. static constexpr decltype(auto) apply(optional<T> const& opt, F&& f)
  290. { return static_cast<F&&>(f)(opt.value_); }
  291. template <typename T, typename F>
  292. static constexpr decltype(auto) apply(optional<T>& opt, F&& f)
  293. { return static_cast<F&&>(f)(opt.value_); }
  294. template <typename F>
  295. static constexpr decltype(auto) apply(optional<> const&, F&& f)
  296. { return static_cast<F&&>(f)(); }
  297. };
  298. //////////////////////////////////////////////////////////////////////////
  299. // Searchable
  300. //////////////////////////////////////////////////////////////////////////
  301. namespace detail {
  302. template <bool>
  303. struct optional_find_if {
  304. template <typename T>
  305. static constexpr auto apply(T const&)
  306. { return hana::nothing; }
  307. };
  308. template <>
  309. struct optional_find_if<true> {
  310. template <typename T>
  311. static constexpr auto apply(T&& t)
  312. { return hana::just(static_cast<T&&>(t)); }
  313. };
  314. }
  315. template <>
  316. struct find_if_impl<optional_tag> {
  317. template <typename T, typename Pred>
  318. static constexpr auto apply(hana::optional<T> const& opt, Pred&& pred) {
  319. constexpr bool found = decltype(static_cast<Pred&&>(pred)(opt.value_))::value;
  320. return detail::optional_find_if<found>::apply(opt.value_);
  321. }
  322. template <typename T, typename Pred>
  323. static constexpr auto apply(hana::optional<T>& opt, Pred&& pred) {
  324. constexpr bool found = decltype(static_cast<Pred&&>(pred)(opt.value_))::value;
  325. return detail::optional_find_if<found>::apply(opt.value_);
  326. }
  327. template <typename T, typename Pred>
  328. static constexpr auto apply(hana::optional<T>&& opt, Pred&& pred) {
  329. constexpr bool found = decltype(
  330. static_cast<Pred&&>(pred)(static_cast<T&&>(opt.value_))
  331. )::value;
  332. return detail::optional_find_if<found>::apply(static_cast<T&&>(opt.value_));
  333. }
  334. template <typename Pred>
  335. static constexpr auto apply(hana::optional<> const&, Pred&&)
  336. { return hana::nothing; }
  337. };
  338. template <>
  339. struct any_of_impl<optional_tag> {
  340. template <typename T, typename Pred>
  341. static constexpr auto apply(hana::optional<T> const& opt, Pred&& pred)
  342. { return static_cast<Pred&&>(pred)(opt.value_); }
  343. template <typename Pred>
  344. static constexpr hana::false_ apply(hana::optional<> const&, Pred&&)
  345. { return {}; }
  346. };
  347. BOOST_HANA_NAMESPACE_END
  348. #endif // !BOOST_HANA_OPTIONAL_HPP