You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

108 lines
3.1 KiB

/**
* @file enum_traits.h
*
* Base template for 'enum_traits' which allow us to access static information about an enum.
*/
#pragma once
#include <cstddef>
namespace devilution {
template <typename T>
struct enum_size {
constexpr static const std::size_t value = static_cast<std::size_t>(T::LAST) + 1;
};
template <typename T>
class enum_values {
public:
class Iterator {
typename std::underlying_type<T>::type m_value;
public:
Iterator(typename std::underlying_type<T>::type value)
: m_value(value)
{
}
const T operator*() const
{
return static_cast<T>(m_value);
}
void operator++()
{
m_value++;
}
bool operator!=(Iterator rhs) const
{
return m_value != rhs.m_value;
}
};
};
template <typename T>
typename enum_values<T>::Iterator begin(enum_values<T>)
{
return typename enum_values<T>::Iterator(static_cast<typename std::underlying_type<T>::type>(T::FIRST));
}
template <typename T>
typename enum_values<T>::Iterator end(enum_values<T>)
{
return typename enum_values<T>::Iterator(static_cast<typename std::underlying_type<T>::type>(T::LAST) + 1);
}
template <typename>
struct is_flags_enum : std::false_type {
};
#define use_enum_as_flags(Type) \
template <> \
struct is_flags_enum<Type> : std::true_type { \
};
template <typename EnumType, std::enable_if_t<std::is_enum<EnumType>::value && is_flags_enum<EnumType>::value, bool> = true>
constexpr EnumType operator|(EnumType lhs, EnumType rhs)
{
using T = std::underlying_type_t<EnumType>;
return static_cast<EnumType>(static_cast<T>(lhs) | static_cast<T>(rhs));
}
template <typename EnumType, std::enable_if_t<std::is_enum<EnumType>::value && is_flags_enum<EnumType>::value, bool> = true>
constexpr EnumType operator|=(EnumType &lhs, EnumType rhs)
{
lhs = lhs | rhs;
return lhs;
}
template <typename EnumType, std::enable_if_t<std::is_enum<EnumType>::value && is_flags_enum<EnumType>::value, bool> = true>
constexpr EnumType operator&(EnumType lhs, EnumType rhs)
{
using T = std::underlying_type_t<EnumType>;
return static_cast<EnumType>(static_cast<T>(lhs) & static_cast<T>(rhs));
}
template <typename EnumType, std::enable_if_t<std::is_enum<EnumType>::value && is_flags_enum<EnumType>::value, bool> = true>
constexpr EnumType operator&=(EnumType &lhs, EnumType rhs)
{
lhs = lhs & rhs;
return lhs;
}
template <typename EnumType, std::enable_if_t<std::is_enum<EnumType>::value && is_flags_enum<EnumType>::value, bool> = true>
constexpr EnumType operator~(EnumType value)
{
using T = std::underlying_type_t<EnumType>;
return static_cast<EnumType>(~static_cast<T>(value));
}
template <typename EnumType, std::enable_if_t<std::is_enum<EnumType>::value && is_flags_enum<EnumType>::value, bool> = true>
constexpr bool HasAnyOf(EnumType lhs, EnumType test)
{
return (lhs & test) != static_cast<EnumType>(0); // Some flags enums may not use a None value outside this check so we don't require an EnumType::None definition here.
}
} // namespace devilution