#ifndef RFL_FIELD_HPP_ #define RFL_FIELD_HPP_ #include #include #include #include #include #include "Literal.hpp" #include "default.hpp" #include "internal/Array.hpp" #include "internal/StringLiteral.hpp" #include "internal/to_std_array.hpp" #include "internal/wrap_in_rfl_array_t.hpp" namespace rfl { /// Used to define a field in the NamedTuple. template struct Field { /// The underlying type. using Type = internal::wrap_in_rfl_array_t; /// The name of the field. using Name = rfl::Literal<_name>; Field(const Type& _value) : value_(_value) {} Field(Type&& _value) noexcept : value_(std::move(_value)) {} Field(Field<_name, T>&& _field) noexcept = default; Field(const Field<_name, T>& _field) = default; template Field(const Field<_name, U>& _field) : value_(_field.get()) {} template Field(Field<_name, U>&& _field) : value_(_field.get()) {} template < class U, typename std::enable_if, bool>::type = true> Field(const U& _value) : value_(_value) {} template < class U, typename std::enable_if, bool>::type = true> Field(U&& _value) noexcept : value_(std::forward(_value)) {} template < class U, typename std::enable_if, bool>::type = true> Field(const Field<_name, U>& _field) : value_(_field.value()) {} /// Assigns the underlying object to its default value. template < class U = Type, typename std::enable_if, bool>:: type = true> Field(const Default& _default) : value_(Type()) {} ~Field() = default; /// The name of the field, for internal use. constexpr static const internal::StringLiteral name_ = _name; /// Returns the underlying object. const Type& get() const { return value_; } /// The name of the field. constexpr static std::string_view name() { return name_.string_view(); } /// Returns the underlying object. Type& operator()() { return value_; } /// Returns the underlying object. const Type& operator()() const { return value_; } /// Assigns the underlying object. auto& operator=(const Type& _value) { value_ = _value; return *this; } /// Assigns the underlying object. auto& operator=(Type&& _value) noexcept { value_ = std::move(_value); return *this; } /// Assigns the underlying object. template < class U, typename std::enable_if, bool>::type = true> auto& operator=(const U& _value) { value_ = _value; return *this; } /// Assigns the underlying object to its default value. template < class U = Type, typename std::enable_if, bool>:: type = true> auto& operator=(const Default& _default) { value_ = Type(); return *this; } /// Assigns the underlying object. Field<_name, T>& operator=(const Field<_name, T>& _field) = default; /// Assigns the underlying object. Field<_name, T>& operator=(Field<_name, T>&& _field) = default; /// Assigns the underlying object. template auto& operator=(const Field<_name, U>& _field) { value_ = _field.get(); return *this; } /// Assigns the underlying object. template auto& operator=(Field<_name, U>&& _field) { value_ = std::forward(_field.value_); return *this; } /// Assigns the underlying object. void set(const Type& _value) { value_ = _value; } /// Assigns the underlying object. void set(Type&& _value) { value_ = std::move(_value); } /// Returns the underlying object. Type& value() { return value_; } /// Returns the underlying object. const Type& value() const { return value_; } /// The underlying value. Type value_; }; template inline auto make_field(T&& _value) { using T0 = std::remove_cvref_t; if constexpr (std::is_array_v) { return Field<_name, T0>(internal::Array(std::forward(_value))); } else { return Field<_name, T0>(std::forward(_value)); } } } // namespace rfl #endif // RFL_FIELD_HPP_