const_propogating_ptr.hpp 3.0 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889
  1. //
  2. // const_propogating_ptr.hpp
  3. // pointers
  4. //
  5. // Created by Sam Jaffe on 12/3/16.
  6. //
  7. #pragma once
  8. #include <memory>
  9. #include "detail/compare.hpp"
  10. #include "detail/get_ptr.hpp"
  11. #include "pointer_fwd.hpp"
  12. namespace pointers {
  13. template <typename P>
  14. class const_propogating_ptr
  15. : private detail::get_ptr<P>,
  16. public detail::pointer_compare<const_propogating_ptr<P>> {
  17. public:
  18. using element_type = typename std::pointer_traits<P>::element_type;
  19. using pointer = element_type *;
  20. using reference = element_type &;
  21. using const_pointer = element_type const *;
  22. using const_reference = element_type const &;
  23. const_propogating_ptr() noexcept : _ptr() {}
  24. const_propogating_ptr(P const & p) noexcept(
  25. std::is_nothrow_copy_constructible<P>::value)
  26. : _ptr(p) {}
  27. const_propogating_ptr(P && p) noexcept(
  28. std::is_nothrow_move_constructible<P>::value)
  29. : _ptr(std::move(p)) {}
  30. template <typename Y, typename = typename std::enable_if<
  31. std::is_constructible<P, Y>::value>::type>
  32. const_propogating_ptr(Y const & p) : _ptr(p) {}
  33. template <typename Y, typename = typename std::enable_if<
  34. std::is_constructible<P, Y>::value>::type>
  35. const_propogating_ptr(Y && p) : _ptr(std::forward<Y>(p)) {}
  36. const_propogating_ptr(const_propogating_ptr &) noexcept(
  37. std::is_nothrow_copy_constructible<P>::value) = default;
  38. const_propogating_ptr(const_propogating_ptr &&) noexcept(
  39. std::is_nothrow_move_constructible<P>::value) = default;
  40. const_propogating_ptr(const_propogating_ptr const &) = delete;
  41. const_propogating_ptr & operator=(const_propogating_ptr &) noexcept(
  42. std::is_nothrow_copy_assignable<P>::value) = default;
  43. const_propogating_ptr & operator=(const_propogating_ptr &&) noexcept(
  44. std::is_nothrow_move_assignable<P>::value) = default;
  45. const_propogating_ptr & operator=(const_propogating_ptr const &) = delete;
  46. template <typename Y>
  47. explicit operator const_propogating_ptr<Y>() &
  48. noexcept(std::is_nothrow_constructible<P, Y>::value) {
  49. return _ptr;
  50. }
  51. template <typename Y>
  52. explicit operator const_ptr<Y>() const
  53. noexcept(std::is_nothrow_constructible<P, Y>::value) {
  54. return _ptr;
  55. }
  56. template <typename Y>
  57. explicit operator const_propogating_ptr<Y>() const & = delete;
  58. operator bool() const noexcept { return static_cast<bool>(_ptr); }
  59. reference operator*() noexcept(noexcept(*_ptr)) { return *_ptr; }
  60. pointer get() noexcept(noexcept(detail::get_ptr<P>::get(_ptr))) {
  61. return detail::get_ptr<P>::get(_ptr);
  62. }
  63. pointer operator->() noexcept(noexcept(get())) { return get(); }
  64. const_reference operator*() const noexcept(noexcept(*_ptr)) {
  65. return *_ptr;
  66. }
  67. const_pointer get() const
  68. noexcept(noexcept(detail::get_ptr<P>::get(_ptr))) {
  69. return detail::get_ptr<P>::get(_ptr);
  70. }
  71. const_pointer operator->() const noexcept(noexcept(get())) { return get(); }
  72. private:
  73. P _ptr;
  74. };
  75. }