xlnt - community edition
optional.hpp
1 // Copyright (c) 2016-2022 Thomas Fussell
2 // Copyright (c) 2024-2025 xlnt-community
3 //
4 // Permission is hereby granted, free of charge, to any person obtaining a copy
5 // of this software and associated documentation files (the "Software"), to deal
6 // in the Software without restriction, including without limitation the rights
7 // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
8 // copies of the Software, and to permit persons to whom the Software is
9 // furnished to do so, subject to the following conditions:
10 //
11 // The above copyright notice and this permission notice shall be included in
12 // all copies or substantial portions of the Software.
13 //
14 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
17 // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
18 // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
19 // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
20 // THE SOFTWARE
21 //
22 // @license: http://www.opensource.org/licenses/mit-license.php
23 // @author: see AUTHORS file
24 
25 #pragma once
26 
27 #include "xlnt/utils/exceptions.hpp"
28 #include "xlnt/utils/numeric.hpp"
29 #include <type_traits>
30 
31 namespace xlnt {
32 
38 template <typename T>
39 class optional
40 {
41 #if ((defined(_MSC_VER) && _MSC_VER <= 1900) || (defined(__GNUC__) && __GNUC__ < 5))
42 // Disable enhanced type checking on Visual Studio <= 2015 and GCC <5
43 #define XLNT_NOEXCEPT_VALUE_COMPAT(...) (false)
44 #else
45 #define XLNT_NOEXCEPT_VALUE_COMPAT(...) (__VA_ARGS__)
46  using ctor_copy_T_noexcept = typename std::conditional<std::is_nothrow_copy_constructible<T>{}, std::true_type, std::false_type>::type;
47  using ctor_move_T_noexcept = typename std::conditional<std::is_nothrow_move_constructible<T>{}, std::true_type, std::false_type>::type;
48  using copy_ctor_noexcept = ctor_copy_T_noexcept;
49  using move_ctor_noexcept = ctor_move_T_noexcept;
50  using set_copy_noexcept_t = typename std::conditional<std::is_nothrow_copy_constructible<T>{} && std::is_nothrow_assignable<T, T>{}, std::true_type, std::false_type>::type;
51  using set_move_noexcept_t = typename std::conditional<std::is_nothrow_move_constructible<T>{} && std::is_nothrow_move_assignable<T>{}, std::true_type, std::false_type>::type;
52  using clear_noexcept_t = typename std::conditional<std::is_nothrow_destructible<T>{}, std::true_type, std::false_type>::type;
53 #endif
54  template <typename U = T, typename std::enable_if<!std::is_floating_point<U>::value>::type * = nullptr>
58  constexpr bool compare_equal(const U &lhs, const U &rhs) const
59  {
60  return lhs == rhs;
61  }
62 
66  template <typename U = T, typename std::enable_if<std::is_floating_point<U>::value>::type * = nullptr>
67  constexpr bool compare_equal(const U &lhs, const U &rhs) const
68  {
69  return detail::float_equals(lhs, rhs);
70  }
71 
72 public:
76  optional() noexcept
77  : has_value_(false)
78  {
79  }
80 
85  optional(const T &value) noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(ctor_copy_T_noexcept{}))
86  : has_value_(true)
87  {
88  new (&storage_) T(value);
89  }
90 
95  optional(T &&value) noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(ctor_move_T_noexcept{}))
96  : has_value_(true)
97  {
98  new (&storage_) T(std::move(value));
99  }
100 
105  optional(const optional &other) noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(copy_ctor_noexcept{}))
106  : has_value_(other.has_value_)
107  {
108  if (has_value_)
109  {
110  new (&storage_) T(other.value_ref());
111  }
112  }
113 
118  optional(optional &&other) noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(move_ctor_noexcept{}))
119  : has_value_(other.has_value_)
120  {
121  if (has_value_)
122  {
123  new (&storage_) T(std::move(other.value_ref()));
124  other.clear();
125  }
126  }
127 
132  optional &operator=(const optional &other) noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(set_copy_noexcept_t{} && clear_noexcept_t{}))
133  {
134  if (other.has_value_)
135  {
136  set(other.value_ref());
137  }
138  else
139  {
140  clear();
141  }
142  return *this;
143  }
144 
149  optional &operator=(optional &&other) noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(set_move_noexcept_t{} && clear_noexcept_t{}))
150  {
151  if (other.has_value_)
152  {
153  set(std::move(other.value_ref()));
154  other.clear();
155  }
156  else
157  {
158  clear();
159  }
160  return *this;
161  }
162 
166  ~optional() noexcept // note:: unconditional because msvc freaks out otherwise
167  {
168  clear();
169  }
170 
175  bool is_set() const noexcept
176  {
177  return has_value_;
178  }
179 
183  void set(const T &value) noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(set_copy_noexcept_t{}))
184  {
185 #if defined(__GNUC__) && !defined(__clang__)
186 #pragma GCC diagnostic push
187 #pragma GCC diagnostic ignored "-Wmaybe-uninitialized"
188 #endif
189  if (has_value_)
190  {
191  value_ref() = value;
192  }
193  else
194  {
195  new (&storage_) T(value);
196  has_value_ = true;
197  }
198 #if defined(__GNUC__) && !defined(__clang__)
199 #pragma GCC diagnostic pop
200 #endif
201  }
202 
206  void set(T &&value) noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(set_move_noexcept_t{}))
207  {
208  // note seperate overload for two reasons (as opposed to perfect forwarding)
209  // 1. have to deal with implicit conversions internally with perfect forwarding
210  // 2. have to deal with the noexcept specfiers for all the different variations
211  // overload is just far and away the simpler solution
212 #if defined(__GNUC__) && !defined(__clang__)
213 #pragma GCC diagnostic push
214 #pragma GCC diagnostic ignored "-Wmaybe-uninitialized"
215 #endif
216  if (has_value_)
217  {
218  value_ref() = std::move(value);
219  }
220  else
221  {
222  new (&storage_) T(std::move(value));
223  has_value_ = true;
224  }
225 #if defined(__GNUC__) && !defined(__clang__)
226 #pragma GCC diagnostic pop
227 #endif
228  }
229 
233  optional &operator=(const T &rhs) noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(set_copy_noexcept_t{}))
234  {
235  set(rhs);
236  return *this;
237  }
238 
242  optional &operator=(T &&rhs) noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(set_move_noexcept_t{}))
243  {
244  set(std::move(rhs));
245  return *this;
246  }
247 
251  void clear() noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(clear_noexcept_t{}))
252  {
253  if (has_value_)
254  {
255  reinterpret_cast<T *>(&storage_)->~T();
256  }
257  has_value_ = false;
258  }
259 
264  T &get()
265  {
266  if (!has_value_)
267  {
268  throw invalid_attribute();
269  }
270 
271  return value_ref();
272  }
273 
278  const T &get() const
279  {
280  if (!has_value_)
281  {
282  throw invalid_attribute();
283  }
284 
285  return value_ref();
286  }
287 
293  bool operator==(const optional<T> &other) const noexcept
294  {
295  if (has_value_ != other.has_value_)
296  {
297  return false;
298  }
299  if (!has_value_)
300  {
301  return true;
302  }
303  // equality is overloaded to provide fuzzy equality when T is a fp number
304  return compare_equal(value_ref(), other.value_ref());
305  }
306 
312  bool operator!=(const optional<T> &other) const noexcept
313  {
314  return !operator==(other);
315  }
316 
317 private:
318  // helpers for getting a T out of storage
319  T &value_ref() noexcept
320  {
321  return *reinterpret_cast<T *>(&storage_);
322  }
323 
324  const T &value_ref() const noexcept
325  {
326  return *reinterpret_cast<const T *>(&storage_);
327  }
328 
329  bool has_value_;
330  alignas(T) unsigned char storage_[sizeof(T)];
331 };
332 
333 #ifdef XLNT_NOEXCEPT_VALUE_COMPAT
334 #undef XLNT_NOEXCEPT_VALUE_COMPAT
335 #endif
336 
337 } // namespace xlnt
optional() noexcept
Default contructor. is_set() will be false initially.
Definition: optional.hpp:76
void clear() noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(clear_noexcept_t{}))
After this is called, is_set() will return false until a new value is provided.
Definition: optional.hpp:251
bool is_set() const noexcept
Returns true if this object currently has a value set. This should be called before accessing the val...
Definition: optional.hpp:175
optional & operator=(T &&rhs) noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(set_move_noexcept_t{}))
Assignment operator overload. Equivalent to setting the value using optional::set.
Definition: optional.hpp:242
optional(optional &&other) noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(move_ctor_noexcept{}))
Move constructs this optional from other. Clears the value from other if set noexcept if T move ctor ...
Definition: optional.hpp:118
bool operator==(const optional< T > &other) const noexcept
Returns true if neither this nor other have a value or both have a value and those values are equal a...
Definition: optional.hpp:293
optional(T &&value) noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(ctor_move_T_noexcept{}))
Constructs this optional with a value. noexcept if T move ctor is noexcept
Definition: optional.hpp:95
Enumerates the possible types a cell can be determined by it&#39;s current value.
Definition: cell.hpp:36
Exception when setting a class&#39;s attribute to an invalid value
Definition: exceptions.hpp:239
Many settings in xlnt are allowed to not have a value set. This class encapsulates a value which may ...
Definition: format.hpp:43
optional & operator=(const T &rhs) noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(set_copy_noexcept_t{}))
Assignment operator overload. Equivalent to setting the value using optional::set.
Definition: optional.hpp:233
~optional() noexcept
Destructor cleans up the T instance if set
Definition: optional.hpp:166
optional & operator=(const optional &other) noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(set_copy_noexcept_t{} &&clear_noexcept_t{}))
Copy assignment of this optional from other noexcept if set and clear are noexcept for T& ...
Definition: optional.hpp:132
optional(const T &value) noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(ctor_copy_T_noexcept{}))
Constructs this optional with a value. noexcept if T copy ctor is noexcept
Definition: optional.hpp:85
optional(const optional &other) noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(copy_ctor_noexcept{}))
Copy constructs this optional from other noexcept if T copy ctor is noexcept
Definition: optional.hpp:105
optional & operator=(optional &&other) noexcept(XLNT_NOEXCEPT_VALUE_COMPAT(set_move_noexcept_t{} &&clear_noexcept_t{}))
Move assignment of this optional from other noexcept if set and clear are noexcept for T&& ...
Definition: optional.hpp:149
bool operator!=(const optional< T > &other) const noexcept
Returns false if neither this nor other have a value or both have a value and those values are equal ...
Definition: optional.hpp:312