File: Flatten.hpp

package info (click to toggle)
reflect-cpp 0.18.0%2Bds-3
  • links: PTS, VCS
  • area: main
  • in suites: trixie
  • size: 12,524 kB
  • sloc: cpp: 44,484; python: 131; makefile: 30; sh: 3
file content (120 lines) | stat: -rw-r--r-- 3,025 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
#ifndef RFL_FLATTEN_HPP_
#define RFL_FLATTEN_HPP_

#include <algorithm>
#include <string_view>
#include <tuple>
#include <type_traits>
#include <utility>

namespace rfl {

/// Used to embed another struct into the generated output.
template <class T>
struct Flatten {
  /// The underlying type.
  using Type = std::remove_cvref_t<T>;

  Flatten() = default;

  Flatten(const Type& _value) : value_(_value) {}

  Flatten(Type&& _value) noexcept : value_(std::forward<Type>(_value)) {}

  Flatten(const Flatten<T>& _f) = default;

  Flatten(Flatten<T>&& _f) noexcept = default;

  template <class U>
  Flatten(const Flatten<U>& _f) : value_(_f.get()) {}

  template <class U>
  Flatten(Flatten<U>&& _f) : value_(_f.get()) {}

  template <class U, typename std::enable_if<std::is_convertible_v<U, Type>,
                                             bool>::type = true>
  Flatten(const U& _value) : value_(_value) {}

  template <class U, typename std::enable_if<std::is_convertible_v<U, Type>,
                                             bool>::type = true>
  Flatten(U&& _value) : value_(_value) {}

  ~Flatten() = default;

  /// Returns the underlying object.
  Type& get() { return value_; }

  /// Returns the underlying object.
  const Type& get() const { return value_; }

  /// Returns the underlying object.
  Type& operator()() { return value_; }

  /// Returns the underlying object.
  const Type& operator()() const { return value_; }

  /// Assigns the underlying object.
  Flatten<T>& operator=(const T& _value) {
    value_ = _value;
    return *this;
  }

  /// Assigns the underlying object.
  Flatten<T>& operator=(T&& _value) {
    value_ = std::forward<Type>(_value);
    return *this;
  }

  /// Assigns the underlying object.
  template <class U, typename std::enable_if<std::is_convertible_v<U, Type>,
                                             bool>::type = true>
  Flatten<T>& operator=(const U& _value) {
    value_ = _value;
    return *this;
  }

  /// Assigns the underlying object.
  Flatten<T>& operator=(const Flatten<T>& _f) = default;

  /// Assigns the underlying object.
  Flatten<T>& operator=(Flatten<T>&& _f) = default;

  /// Assigns the underlying object.
  template <class U>
  Flatten<T>& operator=(const Flatten<U>& _f) {
    value_ = _f.get();
    return *this;
  }

  /// Assigns the underlying object.
  template <class U>
  Flatten<T>& operator=(Flatten<U>&& _f) {
    value_ = std::forward<U>(_f);
    return *this;
  }

  /// Three-way comparison operator
  template <class U>
  auto operator<=>(const Flatten<U>& _f) const {
    return value_ <=> _f.value_;
  }

  /// Equality comparison operator.
  template <class U>
  bool operator==(const Flatten<U>& _f) const {
    return value_ == _f.get();
  }

  /// Assigns the underlying object.
  void set(const Type& _value) { value_ = _value; }

  /// Assigns the underlying object.
  void set(Type&& _value) { value_ = std::forward<Type>(_value); }

  /// The underlying value.
  Type value_;
};

}  // namespace rfl

#endif