codac 1.5.6
Loading...
Searching...
No Matches
codac2_Paving.h
Go to the documentation of this file.
1
9
10#pragma once
11
12#include <map>
13#include <memory>
14#include "codac2_PavingNode.h"
15
16namespace codac2
17{
18 template<typename P>
19 class PavingNode;
20
21 template<typename P>
22 class Subpaving;
23
24 template<typename P,typename... X>
25 requires (std::is_same_v<X,IntervalVector> && ...)
26 class Paving : public Domain
27 {
28 public:
29
30 using Node_ = std::shared_ptr<const PavingNode<P>>;
31 using NodeTuple_ = std::tuple<X...>;
32 using NodeValue_ = std::function<std::list<IntervalVector>(Node_)>;
33 using ConnectedSubset_ = Subpaving<P>;
34
35 Paving(Index n)
36 : Paving(IntervalVector(n))
37 {
38 assert_release(n > 0);
39 }
40
41 Paving(const IntervalVector& x)
42 : _tree(std::make_shared<PavingNode<P>>(*static_cast<P*>(this), x))
43 { }
44
45 Index size() const
46 {
47 return std::get<0>(_tree->boxes()).size();
48 }
49
50 std::shared_ptr<const PavingNode<P>> tree() const
51 {
52 return _tree;
53 }
54
55 std::shared_ptr<PavingNode<P>> tree()
56 {
57 return std::const_pointer_cast<PavingNode<P>>(const_cast<const Paving<P,X...>*>(this)->tree());
58 }
59
60 std::list<IntervalVector> intersecting_boxes(const IntervalVector& x, const NodeValue_& node_value) const
61 {
62 std::list<IntervalVector> l;
63
64 this->tree()->visit([&]
65 (Node_ n)
66 {
67 for(const auto& bi : node_value(n))
68 {
69 auto inter = bi & x;
70 if(bi.intersects(x))
71 l.push_back(bi);
72 }
73
74 return n->hull().intersects(x);
75 });
76
77 return l;
78 }
79
80 std::list<ConnectedSubset_> connected_subsets(const NodeValue_& node_value) const
81 {
82 return connected_subsets(IntervalVector(size()), node_value);
83 }
84
85 std::list<ConnectedSubset_> connected_subsets(const IntervalVector& x0, const NodeValue_& node_value) const
86 {
87 std::list<IntervalVector> l_boxes = intersecting_boxes(x0, node_value);
88 [[maybe_unused]] Index nb_boxes = l_boxes.size();
89 std::list<ConnectedSubset_> l_subsets;
90
91 while(!l_boxes.empty())
92 {
93 auto current_box = l_boxes.front();
94 l_subsets.push_back({ current_box });
95 l_boxes.pop_front();
96
97 std::list<IntervalVector> l_neighbouring_boxes_to_visit { current_box };
98
99 do
100 {
101 current_box = l_neighbouring_boxes_to_visit.front();
102 l_neighbouring_boxes_to_visit.pop_front();
103
104 for(const auto& ni : intersecting_boxes(current_box, node_value))
105 {
106 if(std::find(l_boxes.begin(), l_boxes.end(), ni) != l_boxes.end())
107 {
108 l_boxes.remove(ni);
109 l_neighbouring_boxes_to_visit.push_back(ni);
110 l_subsets.back().push_back(ni);
111 }
112 }
113
114 } while(!l_neighbouring_boxes_to_visit.empty());
115 }
116
117 assert(l_boxes.empty() && "all the nodes should have been visited");
118 assert([&]() -> bool { Index s = 0; for(const auto& si : l_subsets) s += si.size(); return s == nb_boxes; } ()
119 && "the total number of boxes should match the sum of number of boxes of each subset");
120
121 return l_subsets;
122 }
123
124 protected:
125
126 friend class PavingNode<P>;
127
128 static NodeTuple_ init_tuple(const IntervalVector& x)
129 {
130 return std::make_tuple(((X)x)...);
131 }
132
133 std::shared_ptr<PavingNode<P>> _tree; // must be a shared_ptr to allow enable_shared_from_this
134 };
135
136
137 class PavingOut;
138 using PavingOut_Node = PavingNode<PavingOut>;
139
140 class PavingOut : public Paving<PavingOut,IntervalVector>
141 {
142 public:
143
144 PavingOut(Index n);
145 PavingOut(const IntervalVector& x);
146
147 std::list<PavingOut::ConnectedSubset_> connected_subsets(const PavingOut::NodeValue_& node_value = PavingOut::outer) const;
148 std::list<PavingOut::ConnectedSubset_> connected_subsets(const IntervalVector& x0, const PavingOut::NodeValue_& node_value = PavingOut::outer) const;
149
150 static const NodeValue_ outer, outer_complem;
151 };
152
153
154 class PavingInOut;
155 using PavingInOut_Node = PavingNode<PavingInOut>;
156
157 class PavingInOut : public Paving<PavingInOut,IntervalVector,IntervalVector>
158 {
159 public:
160
161 PavingInOut(Index n);
162 PavingInOut(const IntervalVector& x);
163
164 std::list<PavingInOut::ConnectedSubset_> connected_subsets(const PavingInOut::NodeValue_& node_value = PavingInOut::outer) const;
165 std::list<PavingInOut::ConnectedSubset_> connected_subsets(const IntervalVector& x0, const PavingInOut::NodeValue_& node_value = PavingInOut::outer) const;
166
167 static const NodeValue_ outer, outer_complem, inner, bound, all;
168 };
169}