mirror of
https://github.com/minetest/minetest.git
synced 2024-12-26 07:57:29 +01:00
160 lines
4.0 KiB
C++
160 lines
4.0 KiB
C++
/*
|
|
Minetest
|
|
Copyright (C) 2021 rubenwardy
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU Lesser General Public License as published by
|
|
the Free Software Foundation; either version 2.1 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public License along
|
|
with this program; if not, write to the Free Software Foundation, Inc.,
|
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <utility>
|
|
#include "debug.h"
|
|
|
|
struct nullopt_t
|
|
{
|
|
};
|
|
constexpr nullopt_t nullopt{};
|
|
|
|
/**
|
|
* An implementation of optional for C++11, which aims to be
|
|
* compatible with a subset of std::optional features.
|
|
*
|
|
* Unfortunately, Minetest doesn't use C++17 yet.
|
|
*
|
|
* @tparam T The type to be stored
|
|
*/
|
|
template <typename T>
|
|
class Optional
|
|
{
|
|
bool m_has_value = false;
|
|
T m_value;
|
|
|
|
public:
|
|
Optional() noexcept {}
|
|
Optional(nullopt_t) noexcept {}
|
|
|
|
Optional(const T &value) noexcept : m_has_value(true), m_value(value) {}
|
|
Optional(T &&value) noexcept : m_has_value(true), m_value(std::move(value)) {}
|
|
|
|
Optional(const Optional<T> &other) noexcept :
|
|
m_has_value(other.m_has_value), m_value(other.m_value)
|
|
{}
|
|
Optional(Optional<T> &&other) noexcept :
|
|
m_has_value(other.m_has_value), m_value(std::move(other.m_value))
|
|
{
|
|
other.m_has_value = false;
|
|
}
|
|
|
|
Optional<T> &operator=(nullopt_t) noexcept { m_has_value = false; return *this; }
|
|
|
|
Optional<T> &operator=(const Optional<T> &other) noexcept
|
|
{
|
|
if (&other == this)
|
|
return *this;
|
|
m_has_value = other.m_has_value;
|
|
m_value = other.m_value;
|
|
return *this;
|
|
}
|
|
|
|
Optional<T> &operator=(Optional<T> &&other) noexcept
|
|
{
|
|
if (&other == this)
|
|
return *this;
|
|
m_has_value = other.m_has_value;
|
|
m_value = std::move(other.m_value);
|
|
other.m_has_value = false;
|
|
return *this;
|
|
}
|
|
|
|
T &value()
|
|
{
|
|
FATAL_ERROR_IF(!m_has_value, "optional doesn't have value");
|
|
return m_value;
|
|
}
|
|
|
|
const T &value() const
|
|
{
|
|
FATAL_ERROR_IF(!m_has_value, "optional doesn't have value");
|
|
return m_value;
|
|
}
|
|
|
|
const T &value_or(const T &def) const { return m_has_value ? m_value : def; }
|
|
|
|
// Unchecked access consistent with std::optional
|
|
T* operator->() { return &m_value; }
|
|
const T* operator->() const { return &m_value; }
|
|
|
|
T& operator*() { return m_value; }
|
|
const T& operator*() const { return m_value; }
|
|
|
|
bool has_value() const noexcept { return m_has_value; }
|
|
|
|
explicit operator bool() const { return m_has_value; }
|
|
};
|
|
|
|
template <typename T>
|
|
constexpr bool operator==(const Optional<T> &opt, nullopt_t)
|
|
{
|
|
return !opt.has_value();
|
|
}
|
|
template <typename T>
|
|
constexpr bool operator==(nullopt_t, const Optional<T> &opt)
|
|
{
|
|
return !opt.has_value();
|
|
}
|
|
template <typename T>
|
|
constexpr bool operator!=(const Optional<T> &opt, nullopt_t)
|
|
{
|
|
return opt.has_value();
|
|
}
|
|
template <typename T>
|
|
constexpr bool operator!=(nullopt_t, const Optional<T> &opt)
|
|
{
|
|
return opt.has_value();
|
|
}
|
|
|
|
template <typename T, typename U>
|
|
constexpr bool operator==(const Optional<T> &opt, const U &value)
|
|
{
|
|
return opt.has_value() && *opt == value;
|
|
}
|
|
template <typename T, typename U>
|
|
constexpr bool operator==(const T &value, const Optional<U> &opt)
|
|
{
|
|
return opt.has_value() && value == *opt;
|
|
}
|
|
template <typename T, typename U>
|
|
constexpr bool operator!=(const Optional<T> &opt, const U &value)
|
|
{
|
|
return !opt.has_value() || *opt != value;
|
|
}
|
|
template <typename T, typename U>
|
|
constexpr bool operator!=(const T &value, const Optional<U> &opt)
|
|
{
|
|
return !opt.has_value() || value != *opt;
|
|
}
|
|
|
|
|
|
template <typename T, typename U>
|
|
constexpr bool operator==(const Optional<T> &lhs, const Optional<U> &rhs)
|
|
{
|
|
return lhs.has_value() ? *lhs == rhs : nullopt == rhs;
|
|
}
|
|
template <typename T, typename U>
|
|
constexpr bool operator!=(const Optional<T> &lhs, const Optional<U> &rhs)
|
|
{
|
|
return lhs.has_value() ? *lhs != rhs : nullopt != rhs;
|
|
}
|