PiecewiseScalarField3.hh
Go to the documentation of this file.
1/*
2 * Copyright (C) 2022 Open Source Robotics Foundation
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 *
16*/
17#ifndef GZ_MATH_PIECEWISE_SCALAR_FIELD3_HH_
18#define GZ_MATH_PIECEWISE_SCALAR_FIELD3_HH_
19
20#include <algorithm>
21#include <limits>
22#include <sstream>
23#include <utility>
24#include <vector>
25
26#include <gz/math/Region3.hh>
27#include <gz/math/Vector3.hh>
28#include <gz/math/config.hh>
29#include <gz/math/detail/Error.hh>
30
31namespace gz::math
32{
33 // Inline bracket to help doxygen filtering.
34 inline namespace GZ_MATH_VERSION_NAMESPACE {
35 //
39
58 template<typename ScalarField3T, typename ScalarT>
60 {
67
69 public: PiecewiseScalarField3() = default;
70
75 : pieces(_pieces)
76 {
77 for (size_t i = 0; i < pieces.size(); ++i)
78 {
79 if (pieces[i].region.Empty())
80 {
82 errStream << "Region #" << i << " (" << pieces[i].region
83 << ") in piecewise scalar field definition is empty.";
84 detail::LogErrorMessage(errStream.str());
85 }
86 for (size_t j = i + 1; j < pieces.size(); ++j)
87 {
88 if (pieces[i].region.Intersects(pieces[j].region))
89 {
91 errStream << "Detected overlap between regions in "
92 << "piecewise scalar field definition: "
93 << "region #" << i << " (" << pieces[i].region
94 << ") overlaps with region #" << j << " ("
95 << pieces[j].region << "). Region #" << i
96 << " will take precedence when overlapping.";
97 detail::LogErrorMessage(errStream.str());
98 }
99 }
100 }
101 }
102
111
116 public: ScalarT Evaluate(const Vector3<ScalarT> &_p) const
117 {
118 auto it = std::find_if(
119 this->pieces.begin(), this->pieces.end(),
120 [&](const Piece &piece)
121 {
122 return piece.region.Contains(_p);
123 });
124 if (it == this->pieces.end())
125 {
127 }
128 return it->field(_p);
129 }
130
136 public: ScalarT operator()(const Vector3<ScalarT> &_p) const
137 {
138 return this->Evaluate(_p);
139 }
140
152 {
153 if (this->pieces.empty())
154 {
157 }
159 for (const Piece &piece : this->pieces)
160 {
161 if (!piece.region.Empty())
162 {
164 const ScalarT y = piece.field.Minimum(piece.region, p);
165 if (y < yMin)
166 {
167 _pMin = p;
168 yMin = y;
169 }
170 }
171 }
172 return yMin;
173 }
174
179 public: ScalarT Minimum() const
180 {
182 return this->Minimum(pMin);
183 }
184
189 public: friend std::ostream &operator<<(
193 {
194 if (_field.pieces.empty())
195 {
196 return _out << "undefined";
197 }
198 for (size_t i = 0; i < _field.pieces.size() - 1; ++i)
199 {
200 _out << _field.pieces[i].field << " if (x, y, z) in "
201 << _field.pieces[i].region << "; ";
202 }
203 return _out << _field.pieces.back().field
204 << " if (x, y, z) in "
205 << _field.pieces.back().region;
206 }
207
209 private: std::vector<Piece> pieces;
210 };
211
212 template<typename ScalarField3T>
214 template<typename ScalarField3T>
216 } // namespace GZ_MATH_VERSION_NAMESPACE
217} // namespace gz::math
218#endif // GZ_MATH_PIECEWISE_SCALAR_FIELD3_HH_