const_propogating_ptr.hpp 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778
  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 "pointer_fwd.hpp"
  10. #include "ptr_compare.hpp"
  11. template <typename P> class const_propogating_ptr : private detail::get_ptr<P> {
  12. public:
  13. using element_type = typename std::pointer_traits<P>::element_type;
  14. using pointer = element_type *;
  15. using reference = element_type &;
  16. using const_pointer = element_type const *;
  17. using const_reference = element_type const &;
  18. const_propogating_ptr() noexcept : _ptr() {}
  19. const_propogating_ptr(P const & p) noexcept(detail::is_nt_cc<P>::value)
  20. : _ptr(p) {}
  21. const_propogating_ptr(P && p) noexcept(detail::is_nt_mc<P>::value)
  22. : _ptr(std::move(p)) {}
  23. template <typename Y, typename = typename std::enable_if<
  24. std::is_constructible<P, Y>::value>::type>
  25. const_propogating_ptr(Y const & p) : _ptr(p) {}
  26. template <typename Y, typename = typename std::enable_if<
  27. std::is_constructible<P, Y>::value>::type>
  28. const_propogating_ptr(Y && p) : _ptr(std::forward<Y>(p)) {}
  29. const_propogating_ptr(const_propogating_ptr &) noexcept(
  30. detail::is_nt_cc<P>::value) = default;
  31. const_propogating_ptr(const_propogating_ptr &&) noexcept(
  32. detail::is_nt_mc<P>::value) = default;
  33. const_propogating_ptr(const_propogating_ptr const &) = delete;
  34. const_propogating_ptr & operator=(const_propogating_ptr &) noexcept(
  35. detail::is_nt_ca<P>::value) = default;
  36. const_propogating_ptr & operator=(const_propogating_ptr &&) noexcept(
  37. detail::is_nt_ma<P>::value) = default;
  38. const_propogating_ptr & operator=(const_propogating_ptr const &) = delete;
  39. template <typename Y>
  40. explicit operator const_propogating_ptr<Y>() &
  41. noexcept(detail::is_nt_c<P, Y>::value) {
  42. return _ptr;
  43. }
  44. template <typename Y>
  45. explicit operator const_ptr<Y>() const
  46. noexcept(detail::is_nt_c<P, Y>::value) {
  47. return _ptr;
  48. }
  49. template <typename Y>
  50. explicit operator const_propogating_ptr<Y>() const & = delete;
  51. operator bool() const noexcept { return static_cast<bool>(_ptr); }
  52. reference operator*() noexcept(noexcept(*_ptr)) { return *_ptr; }
  53. pointer get() noexcept(noexcept(detail::get_ptr<P>::get(_ptr))) {
  54. return detail::get_ptr<P>::get(_ptr);
  55. }
  56. pointer operator->() noexcept(noexcept(get())) { return get(); }
  57. const_reference operator*() const noexcept(noexcept(*_ptr)) { return *_ptr; }
  58. const_pointer get() const noexcept(noexcept(detail::get_ptr<P>::get(_ptr))) {
  59. return detail::get_ptr<P>::get(_ptr);
  60. }
  61. const_pointer operator->() const noexcept(noexcept(get())) { return get(); }
  62. private:
  63. P _ptr;
  64. };
  65. POINTER_TEMPLATE_COMPARE(const_propogating_ptr)