// file : butl/optional -*- C++ -*- // copyright : Copyright (c) 2014-2017 Code Synthesis Ltd // license : MIT; see accompanying LICENSE file #ifndef BUTL_OPTIONAL #define BUTL_OPTIONAL #include <utility> // move() #include <functional> // hash namespace butl { // Simple optional class template while waiting for std::optional. // struct nullopt_t {constexpr explicit nullopt_t (int) {}}; constexpr const nullopt_t nullopt (1); template <typename T> class optional { public: typedef T value_type; constexpr optional (): value_ (), null_ (true) {} // VC14 needs value_(). constexpr optional (nullopt_t): value_ (), null_ (true) {} constexpr optional (const T& v): value_ (v), null_ (false) {} constexpr optional (T&& v): value_ (std::move (v)), null_ (false) {} optional& operator= (nullopt_t) {value_ = T (); null_ = true; return *this;} optional& operator= (const T& v) {value_ = v; null_ = false; return *this;} optional& operator= (T&& v) {value_ = std::move (v); null_ = false; return *this;} T& value () {return value_;} const T& value () const {return value_;} T* operator-> () {return &value_;} const T* operator-> () const {return &value_;} T& operator* () {return value_;} const T& operator* () const {return value_;} bool has_value () const {return !null_;} explicit operator bool () const {return !null_;} private: T value_; bool null_; }; template <typename T> inline auto operator== (const optional<T>& x, const optional<T>& y) -> decltype (*x == *y) { return static_cast<bool> (x) == static_cast<bool> (y) && (!x || *x == *y); } template <typename T> inline auto operator!= (const optional<T>& x, const optional<T>& y) -> decltype (x == y) { return !(x == y); } template <typename T> inline auto operator< (const optional<T>& x, const optional<T>& y) -> decltype (*x < *y) { bool px (x), py (y); return px < py || (px && py && *x < *y); } template <typename T> inline auto operator> (const optional<T>& x, const optional<T>& y) -> decltype (*x > *y) { return y < x; } } namespace std { template <typename T> struct hash<butl::optional<T>>: hash<T> { using argument_type = butl::optional<T>; size_t operator() (const butl::optional<T>& o) const noexcept (noexcept (hash<T> {} (*o))) { return o ? hash<T>::operator() (*o) : static_cast<size_t> (-3333); } }; } #endif // BUTL_OPTIONAL