38 template<
typename... TYPES>
40 static constexpr u_integer SIZE =
sizeof...(TYPES);
56 template<u_
integer I,
typename T>
64 explicit tupleImpl(
const T& cur = T{});
66 tupleImpl(tupleImpl<I, T>&& other)
noexcept;
68 tupleImpl(
const tupleImpl<I, T>& other);
70 tupleImpl<I, T>&
operator=(tupleImpl<I, T>&& other)
noexcept;
72 template<u_
integer I_DIFF>
75 template<u_
integer I_DIFF,
typename E>
81 std::string
toString(
bool enter)
const override;
84 template<u_
integer I,
typename T,
typename TS>
87 tupleImpl<I + 1, TS> next;
94 explicit tupleImpl(
const T& cur = T{},
const TS& next_elems = TS{});
96 tupleImpl(tupleImpl<I, T, TS>&& other)
noexcept;
98 tupleImpl(
const tupleImpl<I, T, TS>& other);
100 tupleImpl<I, T, TS>&
operator=(tupleImpl<I, T, TS>&& other)
noexcept;
102 template<u_
integer I_DIFF>
105 template<u_
integer I_DIFF,
typename E>
106 void set(
const E& e);
111 std::string
toString(
bool enter)
const override;
114 template<
u_integer I,
typename T,
typename... TS>
115 class tupleImpl<I, T, TS...> :
public printable,
public comparable<tupleImpl<I, T, TS...>>{
117 tupleImpl<I + 1, TS...> next;
124 explicit tupleImpl(
const T& cur,
const TS&... next_elems);
126 explicit tupleImpl(
const T& cur = T{},
const tupleImpl<I + 1, TS...>& nt = tupleImpl<I + 1, TS...>{});
128 tupleImpl(tupleImpl<I, T, TS...>&& other)
noexcept;
130 tupleImpl(
const tupleImpl<I, T, TS...>& other);
132 tupleImpl<I, T, TS...>&
operator=(tupleImpl<I, T, TS...>&& other)
noexcept;
134 template<u_
integer I_DIFF>
137 template<u_
integer I_DIFF,
typename E>
138 void set(
const E& e);
143 std::string
toString(
bool enter)
const override;
146 tupleImpl<0, TYPES...> elems;
149 auto _slice(std::integer_sequence<u_integer, IDX_S...> indexes, std::integral_constant<u_integer, BEGIN_IDX> begin)
const;
152 tuple<TYPES..., O_TYPES...> _concat(
const tuple<O_TYPES...>& other,
153 std::integer_sequence<u_integer, T_SIZE...> ts,
154 std::integer_sequence<u_integer, O_SIZE...> os)
const;
164 explicit tuple(
const TYPES&... e);
170 tuple(
const tuple& other);
183 tuple(tuple&& other)
noexcept;
190 tuple&
operator=(tuple&& other)
noexcept;
203 template<u_
integer IDX>
214 template<u_
integer IDX,
typename E>
215 void set(
const E& e);
223 template<u_
integer BEGIN_IDX, u_
integer N_ELEMS>
246 std::string
toString(
bool enter)
const override;
254 ~tuple()
override =
default;
263 template<
typename F_TYPE,
typename S_TYPE>
274 template<
typename... L_TYPES,
typename... R_TYPES>
275 friend tuple<L_TYPES..., R_TYPES...>
operator+(
const tuple<L_TYPES...>& lt,
const tuple<R_TYPES...>& rt);
278 template<
typename F_TYPE,
typename S_TYPE>
281 template<
typename... L_TYPES,
typename... R_TYPES>
285template<
typename... TYPES>
286template<original::u_
integer I,
typename T>
290template<
typename... TYPES>
291template<original::u_
integer I,
typename T>
293 : cur_elem(std::move(other.cur_elem)) {}
295template<
typename... TYPES>
296template<original::u_
integer I,
typename T>
298 : cur_elem(other.cur_elem) {}
300template<
typename... TYPES>
301template<original::u_
integer I,
typename T>
306 cur_elem = std::move(other.cur_elem);
310template<
typename... TYPES>
311template<original::u_
integer I,
typename T>
312template<original::u_
integer I_DIFF>
314 staticError<outOfBoundError, (I_DIFF > 0)>{};
319template<
typename... TYPES>
320template<original::u_
integer I,
typename T>
321template<original::u_
integer I_DIFF,
typename E>
323 staticError<outOfBoundError, (I_DIFF > 0)>{};
324 staticError<valueError, !std::is_convertible_v<E, T>>{};
326 cur_elem =
static_cast<T
>(e);
329template<
typename... TYPES>
330template<original::u_
integer I,
typename T>
333 if constexpr (Comparable<T>){
334 if (cur_elem != other.cur_elem)
335 return cur_elem < other.cur_elem ? -1 : 1;
340template<
typename... TYPES>
341template<original::u_
integer I,
typename T>
343 std::stringstream ss;
344 if constexpr (I != 0)
346 ss << formatString(cur_elem);
350template<
typename... TYPES>
351template<original::u_
integer I,
typename T,
typename TS>
353 : cur_elem(cur), next(next_elems) {}
355template<
typename... TYPES>
356template<original::u_
integer I,
typename T,
typename TS>
358 : cur_elem(std::move(other.cur_elem)), next(std::move(other.next)) {}
360template<
typename... TYPES>
361template<original::u_
integer I,
typename T,
typename TS>
363 : cur_elem(other.cur_elem), next(other.next) {}
365template<
typename... TYPES>
366template<original::u_
integer I,
typename T,
typename TS>
368 tupleImpl<I, T, TS>&& other)
noexcept -> tupleImpl<I, T, TS>&
372 cur_elem = std::move(other.cur_elem);
373 next = std::move(other.next);
377template<
typename... TYPES>
378template<original::u_
integer I,
typename T,
typename TS>
379template<original::u_
integer I_DIFF>
381 if constexpr (I_DIFF == 0){
384 return next.template get<I_DIFF - 1>();
388template<
typename... TYPES>
389template<original::u_
integer I,
typename T,
typename TS>
390template<original::u_
integer I_DIFF,
typename E>
392 if constexpr (I_DIFF == 0){
393 staticError<valueError, !std::is_convertible_v<E, T>>{};
395 cur_elem =
static_cast<T
>(e);
397 next.template set<I_DIFF - 1,
E>(e);
401template<
typename... TYPES>
402template<original::u_
integer I,
typename T,
typename TS>
404 const tupleImpl<I, T, TS>& other)
const ->
integer
406 if constexpr (Comparable<T>){
407 if (cur_elem != other.cur_elem)
408 return cur_elem < other.cur_elem ? -1 : 1;
410 return next.compareTo(other.next);
413template<
typename... TYPES>
414template<original::u_
integer I,
typename T,
typename TS>
416 std::stringstream ss;
417 if constexpr (I != 0)
419 ss << formatString(cur_elem);
420 ss << formatString(next);
424template<
typename... TYPES>
427 : cur_elem(cur), next(next_elems...) {}
429template<
typename... TYPES>
432 : cur_elem(cur), next(nt) {}
434template<
typename... TYPES>
437 : cur_elem(std::move(other.cur_elem)), next(std::move(other.next)) {}
439template<
typename... TYPES>
442 : cur_elem(other.cur_elem), next(other.next) {}
444template<
typename... TYPES>
447 tupleImpl<I, T, TS...>&& other)
noexcept -> tupleImpl<I, T, TS...>&
451 cur_elem = std::move(other.cur_elem);
452 next = std::move(other.next);
456template<
typename... TYPES>
458template<original::u_
integer I_DIFF>
460 if constexpr (I_DIFF == 0){
463 return next.template get<I_DIFF - 1>();
467template<
typename... TYPES>
469template<original::u_
integer I_DIFF,
typename E>
471 if constexpr (I_DIFF == 0){
472 staticError<valueError, !std::is_convertible_v<E, T>>{};
474 cur_elem =
static_cast<T
>(e);
476 next.template set<I_DIFF - 1,
E>(e);
480template<
typename... TYPES>
484 if constexpr (Comparable<T>){
485 if (cur_elem != other.cur_elem)
486 return cur_elem < other.cur_elem ? -1 : 1;
488 return next.compareTo(other.next);
491template<
typename... TYPES>
494 std::stringstream ss;
495 if constexpr (I != 0)
497 ss << formatString(cur_elem);
498 ss << formatString(next);
502template<
typename... TYPES>
504auto original::tuple<TYPES...>::_slice(std::integer_sequence<u_integer, IDX_S...>,
505 std::integral_constant<u_integer, BEGIN_IDX>)
const {
506 return tuple<
decltype(this->get<BEGIN_IDX + IDX_S>())...>(this->get<BEGIN_IDX + IDX_S>()...);
509template<
typename... TYPES>
512original::tuple<TYPES...>::_concat(
const tuple<O_TYPES...> &other,
513 std::integer_sequence<u_integer, T_SIZE...>,
514 std::integer_sequence<u_integer, O_SIZE...>)
const {
515 return tuple<TYPES..., O_TYPES...>{this->get<T_SIZE>()..., other.template get<O_SIZE>()...};
518template<
typename... TYPES>
519original::tuple<TYPES...>::tuple() : elems() {}
521template<
typename... TYPES>
522original::tuple<TYPES...>::tuple(
const TYPES&... e) : elems(e...) {}
524template<
typename... TYPES>
525original::tuple<TYPES...>::tuple(
const tuple& other) : elems(other.elems) {}
527template<
typename... TYPES>
529 if (
this == &other)
return *
this;
530 this->elems = other.elems;
534template<
typename... TYPES>
535original::tuple<TYPES...>::tuple(tuple&& other) noexcept
536 : elems(std::move(other.elems)) {}
538template<
typename... TYPES>
540 if (
this == &other)
return *
this;
541 this->elems = std::move(other.elems);
545template<
typename... TYPES>
551template<
typename... TYPES>
552template<original::u_
integer IDX>
557template<
typename... TYPES>
558template<original::u_
integer IDX,
typename E>
563template<
typename... TYPES>
564template<original::u_
integer BEGIN_IDX, original::u_
integer N_ELEMS>
566 constexpr bool out_of_bound = BEGIN_IDX >= SIZE || BEGIN_IDX + N_ELEMS > SIZE;
570 std::make_integer_sequence<u_integer, N_ELEMS>{},
571 std::integral_constant<u_integer, BEGIN_IDX>{}
575template<
typename... TYPES>
577 return elems.compareTo(other.elems);
580template<
typename... TYPES>
582 std::stringstream ss;
584 ss <<
"(" << elems <<
")";
588template<
typename... TYPES>
593template<
typename F_TYPE,
typename S_TYPE>
598template<
typename... L_TYPES,
typename... R_TYPES>
599original::tuple<L_TYPES..., R_TYPES...>
601 return lt._concat(rt,
602 std::make_integer_sequence<
u_integer,
sizeof...(L_TYPES)>{},
603 std::make_integer_sequence<
u_integer,
sizeof...(R_TYPES)>{});
Base class for comparable objects.
Definition comparable.h:31
Container for two heterogeneous elements.
Definition couple.h:34
Base class providing polymorphic string conversion capabilities.
Definition printable.h:25
Compile-time error assertion utility.
Definition error.h:73
A fixed-size heterogeneous container that stores elements of different types.
Definition tuple.h:39
tuple & operator=(const tuple &other)
Copy assignment operator.
Definition tuple.h:528
std::string toString(bool enter) const override
Generate formatted string representation.
Definition tuple.h:581
friend tuple< L_TYPES..., R_TYPES... > operator+(const tuple< L_TYPES... > <, const tuple< R_TYPES... > &rt)
Concatenate two tuples (friend function)
std::string className() const override
Get class name identifier.
Definition tuple.h:589
integer compareTo(const tuple &other) const override
Compare two tuples lexicographically.
Definition tuple.h:576
constexpr u_integer size()
Get the number of elements in the tuple.
Definition tuple.h:546
void set(const E &e)
Set element by index.
Definition tuple.h:559
friend tuple< F_TYPE, S_TYPE > makeTuple(const couple< F_TYPE, S_TYPE > &cp)
Create a tuple from a couple.
auto slice() const
Slice the tuple from a specific index.
Definition tuple.h:565
auto get() const
Get element by index.
Definition tuple.h:553
Interface for objects that can be compared.
Generic pair container implementation.
Main namespace for the project Original.
Definition algorithms.h:21
auto operator+(const iterator< T > &it, integer steps) -> iterator< T > *
Adds a number of steps to the iterator's current position and returns a new iterator.
std::uint32_t u_integer
32-bit unsigned integer type for sizes/indexes
Definition config.h:17
std::int64_t integer
64-bit signed integer type for arithmetic operations
Definition config.h:15
tuple< F_TYPE, S_TYPE > makeTuple(const couple< F_TYPE, S_TYPE > &cp)
constexpr long double E
The mathematical constant E (Euler's number).
Definition maths.h:22
Interface for polymorphic string formatting and output.
Type system foundations and concept definitions.