ORIGINAL
Loading...
Searching...
No Matches
tuple.h
Go to the documentation of this file.
1#ifndef TUPLE_H
2#define TUPLE_H
3
4#include "printable.h"
5#include "comparable.h"
6#include "couple.h"
7#include "types.h"
8
22
23namespace original{
24
38 template<typename... TYPES>
39 class tuple final : public printable, public comparable<tuple<TYPES...>>{
40 static constexpr u_integer SIZE = sizeof...(TYPES);
41
53 template<u_integer I, typename... TS>
54 class tupleImpl;
55
56 template<u_integer I, typename T>
57 class tupleImpl<I, T> : public printable, public comparable<tupleImpl<I, T>>{
58 T cur_elem;
59
64 explicit tupleImpl(const T& cur = T{});
65
66 tupleImpl(tupleImpl<I, T>&& other) noexcept;
67
68 tupleImpl(const tupleImpl<I, T>& other);
69
70 tupleImpl<I, T>& operator=(tupleImpl<I, T>&& other) noexcept;
71
72 template<u_integer I_DIFF>
73 auto get() const;
74
75 template<u_integer I_DIFF, typename E>
76 void set(const E& e);
77 public:
78 friend class tuple;
79 integer compareTo(const tupleImpl<I, T>& other) const override;
80
81 std::string toString(bool enter) const override;
82 };
83
84 template<u_integer I, typename T, typename TS>
85 class tupleImpl<I, T, TS> : public printable, public comparable<tupleImpl<I, T, TS>>{
86 T cur_elem;
87 tupleImpl<I + 1, TS> next;
88
94 explicit tupleImpl(const T& cur = T{}, const TS& next_elems = TS{});
95
96 tupleImpl(tupleImpl<I, T, TS>&& other) noexcept;
97
98 tupleImpl(const tupleImpl<I, T, TS>& other);
99
100 tupleImpl<I, T, TS>& operator=(tupleImpl<I, T, TS>&& other) noexcept;
101
102 template<u_integer I_DIFF>
103 auto get() const;
104
105 template<u_integer I_DIFF, typename E>
106 void set(const E& e);
107 public:
108 friend class tuple;
109 integer compareTo(const tupleImpl<I, T, TS>& other) const override;
110
111 std::string toString(bool enter) const override;
112 };
113
114 template<u_integer I, typename T, typename... TS>
115 class tupleImpl<I, T, TS...> : public printable, public comparable<tupleImpl<I, T, TS...>>{
116 T cur_elem;
117 tupleImpl<I + 1, TS...> next;
118
124 explicit tupleImpl(const T& cur, const TS&... next_elems);
125
126 explicit tupleImpl(const T& cur = T{}, const tupleImpl<I + 1, TS...>& nt = tupleImpl<I + 1, TS...>{});
127
128 tupleImpl(tupleImpl<I, T, TS...>&& other) noexcept;
129
130 tupleImpl(const tupleImpl<I, T, TS...>& other);
131
132 tupleImpl<I, T, TS...>& operator=(tupleImpl<I, T, TS...>&& other) noexcept;
133
134 template<u_integer I_DIFF>
135 auto get() const;
136
137 template<u_integer I_DIFF, typename E>
138 void set(const E& e);
139 public:
140 friend class tuple;
141 integer compareTo(const tupleImpl<I, T, TS...>& other) const override;
142
143 std::string toString(bool enter) const override;
144 };
145
146 tupleImpl<0, TYPES...> elems;
147
148 template<u_integer... IDX_S, u_integer BEGIN_IDX>
149 auto _slice(std::integer_sequence<u_integer, IDX_S...> indexes, std::integral_constant<u_integer, BEGIN_IDX> begin) const;
150
151 template<typename... O_TYPES, u_integer... T_SIZE, u_integer... O_SIZE>
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;
155
156 public:
157
158 explicit tuple();
159
164 explicit tuple(const TYPES&... e);
165
170 tuple(const tuple& other);
171
177 tuple& operator=(const tuple& other);
178
183 tuple(tuple&& other) noexcept;
184
190 tuple& operator=(tuple&& other) noexcept;
191
196 constexpr u_integer size();
197
203 template<u_integer IDX>
204 auto get() const;
205
214 template<u_integer IDX, typename E>
215 void set(const E& e);
216
223 template<u_integer BEGIN_IDX, u_integer N_ELEMS>
224 auto slice() const;
225
239 integer compareTo(const tuple& other) const override;
240
246 std::string toString(bool enter) const override;
247
252 std::string className() const override;
253
254 ~tuple() override = default;
255
263 template<typename F_TYPE, typename S_TYPE>
264 friend tuple<F_TYPE, S_TYPE> makeTuple(const couple<F_TYPE, S_TYPE>& cp);
265
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);
276 };
277
278 template<typename F_TYPE, typename S_TYPE>
280
281 template<typename... L_TYPES, typename... R_TYPES>
282 tuple<L_TYPES..., R_TYPES...> operator+(const tuple<L_TYPES...>& lt, const tuple<R_TYPES...>& rt);
283}
284
285template<typename... TYPES>
286template<original::u_integer I, typename T>
288 : cur_elem(cur) {}
289
290template<typename... TYPES>
291template<original::u_integer I, typename T>
292original::tuple<TYPES...>::tupleImpl<I, T>::tupleImpl(tupleImpl<I, T>&& other) noexcept
293 : cur_elem(std::move(other.cur_elem)) {}
294
295template<typename... TYPES>
296template<original::u_integer I, typename T>
298 : cur_elem(other.cur_elem) {}
299
300template<typename... TYPES>
301template<original::u_integer I, typename T>
302auto original::tuple<TYPES...>::tupleImpl<I, T>::operator=(tupleImpl<I, T>&& other) noexcept -> tupleImpl<I, T>&
303{
304 if (this == &other)
305 return *this;
306 cur_elem = std::move(other.cur_elem);
307 return *this;
308}
309
310template<typename... TYPES>
311template<original::u_integer I, typename T>
312template<original::u_integer I_DIFF>
314 staticError<outOfBoundError, (I_DIFF > 0)>{};
315
316 return cur_elem;
317}
318
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>>{};
325
326 cur_elem = static_cast<T>(e);
327}
328
329template<typename... TYPES>
330template<original::u_integer I, typename T>
331auto original::tuple<TYPES...>::tupleImpl<I, T>::compareTo(const tupleImpl<I, T>& other) const -> integer
332{
333 if constexpr (Comparable<T>){
334 if (cur_elem != other.cur_elem)
335 return cur_elem < other.cur_elem ? -1 : 1;
336 }
337 return 0;
338}
339
340template<typename... TYPES>
341template<original::u_integer I, typename T>
342std::string original::tuple<TYPES...>::tupleImpl<I, T>::toString(bool enter) const {
343 std::stringstream ss;
344 if constexpr (I != 0)
345 ss << ", ";
346 ss << formatString(cur_elem);
347 return ss.str();
348}
349
350template<typename... TYPES>
351template<original::u_integer I, typename T, typename TS>
352original::tuple<TYPES...>::tupleImpl<I, T, TS>::tupleImpl(const T& cur, const TS& next_elems)
353 : cur_elem(cur), next(next_elems) {}
354
355template<typename... TYPES>
356template<original::u_integer I, typename T, typename TS>
357original::tuple<TYPES...>::tupleImpl<I, T, TS>::tupleImpl(tupleImpl<I, T, TS>&& other) noexcept
358 : cur_elem(std::move(other.cur_elem)), next(std::move(other.next)) {}
359
360template<typename... TYPES>
361template<original::u_integer I, typename T, typename TS>
362original::tuple<TYPES...>::tupleImpl<I, T, TS>::tupleImpl(const tupleImpl<I, T, TS>& other)
363 : cur_elem(other.cur_elem), next(other.next) {}
364
365template<typename... TYPES>
366template<original::u_integer I, typename T, typename TS>
368 tupleImpl<I, T, TS>&& other) noexcept -> tupleImpl<I, T, TS>&
369{
370 if (this == &other)
371 return *this;
372 cur_elem = std::move(other.cur_elem);
373 next = std::move(other.next);
374 return *this;
375}
376
377template<typename... TYPES>
378template<original::u_integer I, typename T, typename TS>
379template<original::u_integer I_DIFF>
381 if constexpr (I_DIFF == 0){
382 return cur_elem;
383 } else{
384 return next.template get<I_DIFF - 1>();
385 }
386}
387
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>>{};
394
395 cur_elem = static_cast<T>(e);
396 } else{
397 next.template set<I_DIFF - 1, E>(e);
398 }
399}
400
401template<typename... TYPES>
402template<original::u_integer I, typename T, typename TS>
404 const tupleImpl<I, T, TS>& other) const -> integer
405{
406 if constexpr (Comparable<T>){
407 if (cur_elem != other.cur_elem)
408 return cur_elem < other.cur_elem ? -1 : 1;
409 }
410 return next.compareTo(other.next);
411}
412
413template<typename... TYPES>
414template<original::u_integer I, typename T, typename TS>
416 std::stringstream ss;
417 if constexpr (I != 0)
418 ss << ", ";
419 ss << formatString(cur_elem);
420 ss << formatString(next);
421 return ss.str();
422}
423
424template<typename... TYPES>
425template<original::u_integer I, typename T, typename... TS>
426original::tuple<TYPES...>::tupleImpl<I, T, TS...>::tupleImpl(const T& cur, const TS&... next_elems)
427 : cur_elem(cur), next(next_elems...) {}
428
429template<typename... TYPES>
430template<original::u_integer I, typename T, typename... TS>
431original::tuple<TYPES...>::tupleImpl<I, T, TS...>::tupleImpl(const T& cur, const tupleImpl<I + 1, TS...>& nt)
432 : cur_elem(cur), next(nt) {}
433
434template<typename... TYPES>
435template<original::u_integer I, typename T, typename... TS>
436original::tuple<TYPES...>::tupleImpl<I, T, TS...>::tupleImpl(tupleImpl<I, T, TS...> &&other) noexcept
437 : cur_elem(std::move(other.cur_elem)), next(std::move(other.next)) {}
438
439template<typename... TYPES>
440template<original::u_integer I, typename T, typename... TS>
441original::tuple<TYPES...>::tupleImpl<I, T, TS...>::tupleImpl(const tupleImpl<I, T, TS...>& other)
442 : cur_elem(other.cur_elem), next(other.next) {}
443
444template<typename... TYPES>
445template<original::u_integer I, typename T, typename... TS>
447 tupleImpl<I, T, TS...>&& other) noexcept -> tupleImpl<I, T, TS...>&
448{
449 if (this == &other)
450 return *this;
451 cur_elem = std::move(other.cur_elem);
452 next = std::move(other.next);
453 return *this;
454}
455
456template<typename... TYPES>
457template<original::u_integer I, typename T, typename... TS>
458template<original::u_integer I_DIFF>
460 if constexpr (I_DIFF == 0){
461 return cur_elem;
462 } else{
463 return next.template get<I_DIFF - 1>();
464 }
465}
466
467template<typename... TYPES>
468template<original::u_integer I, typename T, typename... TS>
469template<original::u_integer I_DIFF, typename E>
471 if constexpr (I_DIFF == 0){
472 staticError<valueError, !std::is_convertible_v<E, T>>{};
473
474 cur_elem = static_cast<T>(e);
475 } else{
476 next.template set<I_DIFF - 1, E>(e);
477 }
478}
479
480template<typename... TYPES>
481template<original::u_integer I, typename T, typename... TS>
483original::tuple<TYPES...>::tupleImpl<I, T, TS...>::compareTo(const tupleImpl<I, T, TS...>& other) const {
484 if constexpr (Comparable<T>){
485 if (cur_elem != other.cur_elem)
486 return cur_elem < other.cur_elem ? -1 : 1;
487 }
488 return next.compareTo(other.next);
489}
490
491template<typename... TYPES>
492template<original::u_integer I, typename T, typename... TS>
494 std::stringstream ss;
495 if constexpr (I != 0)
496 ss << ", ";
497 ss << formatString(cur_elem);
498 ss << formatString(next);
499 return ss.str();
500}
501
502template<typename... TYPES>
503template<original::u_integer... IDX_S, original::u_integer BEGIN_IDX>
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>()...);
507}
508
509template<typename... TYPES>
510template<typename... O_TYPES, original::u_integer... T_SIZE, original::u_integer... O_SIZE>
511original::tuple<TYPES..., O_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>()...};
516}
517
518template<typename... TYPES>
519original::tuple<TYPES...>::tuple() : elems() {}
520
521template<typename... TYPES>
522original::tuple<TYPES...>::tuple(const TYPES&... e) : elems(e...) {}
523
524template<typename... TYPES>
525original::tuple<TYPES...>::tuple(const tuple& other) : elems(other.elems) {}
526
527template<typename... TYPES>
529 if (this == &other) return *this;
530 this->elems = other.elems;
531 return *this;
532}
533
534template<typename... TYPES>
535original::tuple<TYPES...>::tuple(tuple&& other) noexcept
536 : elems(std::move(other.elems)) {}
537
538template<typename... TYPES>
539original::tuple<TYPES...>& original::tuple<TYPES...>::operator=(tuple&& other) noexcept {
540 if (this == &other) return *this;
541 this->elems = std::move(other.elems);
542 return *this;
543}
544
545template<typename... TYPES>
547{
548 return SIZE;
549}
550
551template<typename... TYPES>
552template<original::u_integer IDX>
554 return elems.template get<IDX>();
555}
556
557template<typename... TYPES>
558template<original::u_integer IDX, typename E>
560 elems.template set<IDX, E>(e);
561}
562
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;
568
569 return this->_slice(
570 std::make_integer_sequence<u_integer, N_ELEMS>{},
571 std::integral_constant<u_integer, BEGIN_IDX>{}
572 );
573}
574
575template<typename... TYPES>
577 return elems.compareTo(other.elems);
578}
579
580template<typename... TYPES>
581std::string original::tuple<TYPES...>::toString(bool enter) const {
582 std::stringstream ss;
583 ss << this->className();
584 ss << "(" << elems << ")";
585 return ss.str();
586}
587
588template<typename... TYPES>
590 return "tuple";
591}
592
593template<typename F_TYPE, typename S_TYPE>
595 return original::tuple<F_TYPE, S_TYPE>{cp.template get<0>(), cp.template get<1>()};
596}
597
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)>{});
604}
605
606#endif //TUPLE_H
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... > &lt, 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.