6#ifndef DUNE_TYPETREE_TREEPATH_HH
7#define DUNE_TYPETREE_TREEPATH_HH
14#include <dune/common/documentation.hh>
15#include <dune/common/version.hh>
16#include <dune/common/typetraits.hh>
17#include <dune/common/indices.hh>
18#include <dune/common/hybridutilities.hh>
19#include <dune/common/typelist.hh>
27 struct check_size_t_impl
29 static constexpr auto check() {
30 return std::is_same_v<T, std::size_t>;
34 template<
class T, T v>
35 struct check_size_t_impl<std::integral_constant<T,v>>
37 static constexpr auto check() {
38 return std::is_same_v<T, std::size_t>;
43 constexpr auto check_size_t() {
44 return check_size_t_impl<T>::check();
48 constexpr auto cast_size_t(
const T & v) {
51 std::is_convertible_v<T,std::size_t> &&
52 std::is_integral_v<T>,
53 "HybridTreePath indices must be convertible to std::size_t or std::integral_constant<std::size_t,v>");
56 "HybridTreePath indices must be convertible to std::size_t or std::integral_constant<std::size_t,v>");
57 return std::size_t(v);
60 template<
class T, T v>
61 constexpr auto cast_size_t(std::integral_constant<T,v>) {
65 std::is_convertible_v<T,std::size_t> &&
66 std::is_integral_v<T> &&
68 "HybridTreePath indices must be convertible to std::size_t or std::integral_constant<std::size_t,v>");
69 return std::integral_constant<std::size_t,v>();
74 [[deprecated(
"HybridTreePath index storage should be std::size_t or std::integral_constant<std::size_t,v>!\n"
75 "Using anything else is deprecated and will not possible after the 2.10 release.\n"
76 "It is adviced not to specify the template parameters expicitly,\n"
77 "but to use the helper functions `hybridTreePath` or `treePath`."
78 "These take care of converting indices to the appropriate storage.")]]
79 constexpr bool check_storage_type(MetaType<T>) {
84 template<std::
size_t v>
85 constexpr bool check_storage_type(MetaType<std::integral_constant<std::size_t,v>>) {
90 constexpr bool check_storage_type(MetaType<std::size_t>) {
95 template<
typename... T>
102 namespace TreePathType {
103 enum Type { fullyStatic, dynamic };
109 template<
typename,std::
size_t>
110 struct TreePathPushBack;
112 template<
typename,std::
size_t>
113 struct TreePathPushFront;
119 struct TreePathFront;
121 template<
typename, std::size_t...>
122 struct TreePathPopBack;
125 struct TreePathPopFront;
127 template<
typename,
typename>
128 struct TreePathConcat;
130 template<std::size_t... i>
131 void print_tree_path(std::ostream& os)
134 template<std::size_t k, std::size_t... i>
135 void print_tree_path(std::ostream& os)
138 print_tree_path<i...>(os);
153 template<
typename... T>
158#if DUNE_VERSION_GTE(TYPETREE,2,10)
160 static_assert((... && Impl::check_size_t<T>()),
161 "HybridTreePath index storage must be std::size_t or std::integral_constant<std::size_t,v>");
172 [[maybe_unused]]
constexpr bool check =
173 (... && Impl::check_storage_type(MetaType<T>()) );
186 [[maybe_unused]]
constexpr bool check =
187 (... && Impl::check_storage_type(MetaType<T>()) );
191 template<
typename... U,
192 typename std::enable_if_t<(
sizeof...(T) > 0 &&
sizeof...(U) ==
sizeof...(T)),
bool> =
true>
196 [[maybe_unused]]
constexpr bool check =
197 (... && Impl::check_storage_type(MetaType<T>()) );
207 [[nodiscard]]
constexpr static std::size_t
size()
213 [[nodiscard]]
constexpr static std::size_t
max_size()
219 template<std::size_t i,
220 std::enable_if_t<(
sizeof...(T) > i),
bool> =
true>
221 [[nodiscard]]
constexpr auto operator[](Dune::index_constant<i>)
const
223 return std::get<i>(_data);
227 [[nodiscard]]
constexpr std::size_t
operator[](std::size_t pos)
const
229 std::size_t entry = 0;
230 Dune::Hybrid::forEach(
enumerate(), [&] (
auto i) {
238 template<std::size_t i,
239 std::enable_if_t<(
sizeof...(T) > i),
bool> =
true>
240 [[nodiscard]]
constexpr auto element(Dune::index_constant<i> pos = {})
const
242 return std::get<i>(_data);
246 [[nodiscard]]
constexpr std::size_t
element(std::size_t pos)
const
248 std::size_t entry = 0;
249 Dune::Hybrid::forEach(
enumerate(), [&] (
auto i) {
257 template<std::size_t n =
sizeof...(T),
258 std::enable_if_t<(n > 0 && n ==
sizeof...(T)),
bool> =
true>
259 [[nodiscard]]
constexpr auto front()
const
261 return std::get<0>(_data);
265 template<std::size_t n =
sizeof...(T),
266 std::enable_if_t<(n > 0 && n ==
sizeof...(T)),
bool> =
true>
267 [[nodiscard]]
constexpr auto back()
const
269 return std::get<n-1>(_data);
277 using Data = std::tuple<T...>;
291 template<
typename... T>
295 static_assert((... && Impl::check_size_t<T>()),
296 "HybridTreePath indices must be of type std::size_t or std::integral_constant<std::size_t,v>");
308 template<
typename... T>
322 template<
typename... T>
323 [[nodiscard]]
constexpr auto treePath(
const T&... t)
330 template<
typename... T>
353 template<std::size_t i,
typename... T>
355 ->
typename std::decay<
decltype(std::get<i>(tp._data))>::type
357 return std::get<i>(tp._data);
376 template<std::size_t i,
typename... T>
379 return std::get<i>(tp._data);
388 template<
typename... T>
390 ->
decltype(tp.
back())
401 template<
typename... T>
403 ->
decltype(tp.
front())
412 template<
typename... T>
415 return HybridTreePath<T...,std::size_t>(std::tuple_cat(tp._data,std::make_tuple(i)));
433 template<std::size_t i,
typename... T>
436 return HybridTreePath<T...,index_constant<i> >(std::tuple_cat(tp._data,std::make_tuple(i_)));
443 template<
typename... T>
446 return HybridTreePath<std::size_t,T...>(std::tuple_cat(std::make_tuple(element),tp._data));
464 template<std::size_t i,
typename... T>
467 return HybridTreePath<index_constant<i>,T...>(std::tuple_cat(std::make_tuple(_i),tp._data));
482 template<
typename I,
typename... T, std::enable_if_t<(
sizeof...(T) > 0),
bool> =
true>
484 using ::Dune::Hybrid::plus;
501 template<
typename I,
typename... T, std::enable_if_t<(
sizeof...(T) > 0),
bool> =
true>
503 using ::Dune::Hybrid::plus;
508 template<
class... Head,
class... Other>
516 constexpr std::size_t size =
sizeof...(T);
517 return unpackIntegerSequence([&](
auto... i){
518 return treePath(tp[index_constant<size-i-1>{}] ...);
519 }, std::make_index_sequence<size>{});
526 template <
class... T, std::enable_if_t<(
sizeof...(T) > 0),
bool> =
true>
529 return unpackIntegerSequence([&](
auto... i){
530 return HybridTreePath{std::make_tuple(std::get<i+1>(tp._data)...)};
531 }, std::make_index_sequence<(
sizeof...(T) - 1)>{});
538 template <
class... T, std::enable_if_t<(
sizeof...(T) > 0),
bool> =
true>
541 return unpackIntegerSequence([&](
auto... i){
543 }, std::make_index_sequence<(
sizeof...(T) - 1)>{});
555 template <
class... S,
class... T>
560 if constexpr (
sizeof...(S) ==
sizeof...(T)) {
561 if constexpr ((Dune::IsInteroperable<S,T>::value &&...)) {
562 return unpackIntegerSequence([&](
auto... i){
563 return ((std::get<i>(lhs._data) == std::get<i>(rhs._data)) &&...);
564 }, std::make_index_sequence<(
sizeof...(S))>{});
579 template <
class S, S... lhs,
class T, T... rhs>
589 template <
class... S,
class... T>
594 return !(lhs == rhs);
598 template <
class S, S... lhs,
class T, T... rhs>
607 inline namespace Literals {
614 template <
char... digits>
615 constexpr auto operator""_tp()
617 using namespace Dune::Indices::Literals;
624 template<std::size_t... i>
625 struct TreePathSize<HybridTreePath<index_constant<i>...> >
626 :
public index_constant<sizeof...(i)>
630 template<std::size_t k, std::size_t... i>
631 struct TreePathPushBack<HybridTreePath<index_constant<i>...>,k>
633 typedef HybridTreePath<index_constant<i>...,index_constant<k>> type;
636 template<std::size_t k, std::size_t... i>
637 struct TreePathPushFront<HybridTreePath<index_constant<i>...>,k>
639 typedef HybridTreePath<index_constant<k>,index_constant<i>...> type;
642 template<std::
size_t k>
643 struct TreePathBack<HybridTreePath<index_constant<k>>>
644 :
public index_constant<k>
647 template<std::size_t j, std::size_t k, std::size_t... l>
648 struct TreePathBack<HybridTreePath<index_constant<j>,index_constant<k>,index_constant<l>...>>
649 :
public TreePathBack<HybridTreePath<index_constant<k>,index_constant<l>...>>
652 template<std::size_t k, std::size_t... i>
653 struct TreePathFront<HybridTreePath<index_constant<k>,index_constant<i>...>>
654 :
public index_constant<k>
657 template<std::size_t k, std::size_t... i>
658 struct TreePathPopBack<HybridTreePath<index_constant<k>>,i...>
660 typedef HybridTreePath<index_constant<i>...> type;
663 template<std::size_t j,
667 struct TreePathPopBack<HybridTreePath<index_constant<j>,index_constant<k>,index_constant<l>...>,i...>
668 :
public TreePathPopBack<HybridTreePath<index_constant<k>,index_constant<l>...>,i...,j>
671 template<std::size_t k, std::size_t... i>
672 struct TreePathPopFront<HybridTreePath<index_constant<k>,index_constant<i>...> >
674 typedef HybridTreePath<index_constant<i>...> type;
677 template<std::size_t... i, std::size_t... k>
678 struct TreePathConcat<HybridTreePath<index_constant<i>...>,HybridTreePath<index_constant<k>...> >
680 typedef HybridTreePath<index_constant<i>...,index_constant<k>...> type;
688 template<std::size_t i,
typename... T>
689 typename std::enable_if<
692 print_hybrid_tree_path(std::ostream& os,
const HybridTreePath<T...>& tp, index_constant<i> _i)
696 template<std::size_t i,
typename... T>
697 typename std::enable_if<
700 print_hybrid_tree_path(std::ostream& os,
const HybridTreePath<T...>& tp, index_constant<i> _i)
703 print_hybrid_tree_path(os,tp,index_constant<i+1>{});
711 template<
typename... T>
714 os <<
"HybridTreePath< ";
715 impl::print_hybrid_tree_path(os, tp, index_constant<0>{});
720 template<std::size_t... i>
721 using StaticTreePath = HybridTreePath<Dune::index_constant<i>...>;
A hybrid version of TreePath that supports both compile time and run time indices.
Definition: treepath.hh:155
constexpr auto back() const
Get the last index value. Only available in non-empty paths.
Definition: treepath.hh:267
constexpr std::size_t element(std::size_t pos) const
Get the index value at position pos.
Definition: treepath.hh:246
constexpr HybridTreePath(std::tuple< T... > t)
Constructor from a std::tuple
Definition: treepath.hh:183
constexpr HybridTreePath(U... t)
Constructor from arguments.
Definition: treepath.hh:193
constexpr HybridTreePath()
Default constructor.
Definition: treepath.hh:170
constexpr auto element(Dune::index_constant< i > pos={}) const
Get the last index value.
Definition: treepath.hh:240
static constexpr std::size_t size()
Get the size (length) of this path.
Definition: treepath.hh:207
constexpr auto operator[](Dune::index_constant< i >) const
Get the index value at position pos.
Definition: treepath.hh:221
constexpr auto front() const
Get the first index value. Only available in non-empty paths.
Definition: treepath.hh:259
static constexpr index_sequence enumerate()
Returns an index_sequence for enumerating the components of this HybridTreePath.
Definition: treepath.hh:201
constexpr std::size_t operator[](std::size_t pos) const
Get the index value at position pos.
Definition: treepath.hh:227
static constexpr std::size_t max_size()
Get the size (length) of this path.
Definition: treepath.hh:213
std::index_sequence_for< T... > index_sequence
An index_sequence for the entries in this HybridTreePath.
Definition: treepath.hh:167
std::ostream & operator<<(std::ostream &os, const HybridTreePath< T... > &tp)
Dumps a HybridTreePath to a stream.
Definition: treepath.hh:712
constexpr std::size_t treePathSize(const HybridTreePath< T... > &)
Returns the size (number of components) of the given HybridTreePath.
Definition: treepath.hh:331
constexpr auto back(const HybridTreePath< T... > &tp) -> decltype(tp.back())
Returns a copy of the last element of the HybridTreePath.
Definition: treepath.hh:389
constexpr auto accumulate_front(const HybridTreePath< T... > &tp, I i)
Hybrid utility that accumulates to the front of a multi-index.
Definition: treepath.hh:502
constexpr auto pop_front(const HybridTreePath< T... > &tp)
Removes first index on a HybridTreePath.
Definition: treepath.hh:527
constexpr auto reverse(const HybridTreePath< T... > &tp)
Reverses the order of the elements in the path.
Definition: treepath.hh:515
constexpr auto operator!=(const HybridTreePath< std::integral_constant< S, lhs >... > &, const HybridTreePath< std::integral_constant< T, rhs >... > &)
Compare two static HybridTreePaths for unequality.
Definition: treepath.hh:599
constexpr auto accumulate_back(const HybridTreePath< T... > &tp, I i)
Hybrid utility that accumulates to the back of a multi-index.
Definition: treepath.hh:483
constexpr auto join(const HybridTreePath< Head... > &head, const Other &... tail)
Join two tree paths into one.
Definition: treepath.hh:509
constexpr auto treePathEntry(const HybridTreePath< T... > &tp, index_constant< i >={}) -> typename std::decay< decltype(std::get< i >(tp._data))>::type
Returns a copy of the i-th element of the HybridTreePath.
Definition: treepath.hh:354
constexpr auto pop_back(const HybridTreePath< T... > &tp)
Removes last index on a HybridTreePath.
Definition: treepath.hh:539
constexpr auto hybridTreePath(const T &... t)
Constructs a new HybridTreePath from the given indices.
Definition: treepath.hh:309
constexpr std::size_t treePathIndex(const HybridTreePath< T... > &tp, index_constant< i >={})
Returns the index value of the i-th element of the HybridTreePath.
Definition: treepath.hh:377
constexpr auto makeTreePath(const T... t)
helper function to construct a new HybridTreePath from the given indices.
Definition: treepath.hh:292
constexpr auto operator==(const HybridTreePath< std::integral_constant< S, lhs >... > &, const HybridTreePath< std::integral_constant< T, rhs >... > &)
Overload for purely static HybridTreePaths.
Definition: treepath.hh:580
constexpr HybridTreePath< T..., index_constant< i > > push_back(const HybridTreePath< T... > &tp, index_constant< i > i_={})
Appends a compile time index to a HybridTreePath.
Definition: treepath.hh:434
constexpr auto treePath(const T &... t)
Constructs a new HybridTreePath from the given indices.
Definition: treepath.hh:323
constexpr HybridTreePath< index_constant< i >, T... > push_front(const HybridTreePath< T... > &tp, index_constant< i > _i={})
Prepends a compile time index to a HybridTreePath.
Definition: treepath.hh:465
constexpr auto front(const HybridTreePath< T... > &tp) -> decltype(tp.front())
Returns a copy of the first element of the HybridTreePath.
Definition: treepath.hh:402
Statically combine two values of type result_type using +.
Definition: accumulate_static.hh:49