OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
curve.hpp
Go to the documentation of this file.
1#pragma once
2
3#include <ossia/detail/config.hpp>
4
5#include <ossia/detail/flat_map.hpp>
6#include <ossia/detail/optional.hpp>
11#include <ossia/network/base/node.hpp>
12#include <ossia/network/base/parameter.hpp>
14#include <ossia/network/value/value.hpp>
15
16#include <functional>
17#include <memory>
18#include <utility>
19#include <vector>
20
25namespace ossia
26{
27class destination;
28
29template <typename T>
30static constexpr auto curve_segment_type_map()
31{
32 if constexpr(std::is_same_v<T, int>)
33 return ossia::curve_segment_type::INT;
34 else if constexpr(std::is_same_v<T, int64_t>)
35 return ossia::curve_segment_type::INT64;
36 else if constexpr(std::is_same_v<T, float>)
37 return ossia::curve_segment_type::FLOAT;
38 else if constexpr(std::is_same_v<T, double>)
39 return ossia::curve_segment_type::DOUBLE;
40 else if constexpr(std::is_same_v<T, bool>)
41 return ossia::curve_segment_type::BOOL;
42 else if constexpr(std::is_same_v<T, ossia::value>)
43 return ossia::curve_segment_type::ANY;
44 else
45 return nullptr;
46}
47
48template <typename K, typename V>
49using curve_map = ossia::flat_map<K, V>;
50
51template <typename X, typename Y>
58class curve final : public curve_abstract
59{
60public:
61 using abscissa_type = X;
62 using ordinate_type = Y;
63 using curve_type = curve<X, Y>;
64 using map_type = curve_map<X, std::pair<Y, ossia::curve_segment<Y>>>;
65
66 curve() = default;
67 curve(const curve& other)
68 {
69 m_x0 = other.m_x0;
70 m_y0 = other.m_y0;
71 m_y0_destination = other.m_y0_destination;
72
73 m_points = other.m_points;
74
75 m_y0_cacheUsed = false;
76 }
77
78 curve(curve&& other) noexcept
79 {
80 m_x0 = other.m_x0;
81 m_y0 = other.m_y0;
82 m_y0_destination = std::move(other.m_y0_destination);
83
84 m_points = std::move(other.m_points);
85
86 m_y0_cacheUsed = false;
87 }
88
89 curve& operator=(const curve& other)
90 {
91 m_x0 = other.m_x0;
92 m_y0 = other.m_y0;
93 m_y0_destination = other.m_y0_destination;
94
95 m_points = other.m_points;
96
97 m_y0_cacheUsed = false;
98 return *this;
99 }
100 curve& operator=(curve&& other) noexcept
101 {
102 m_x0 = other.m_x0;
103 m_y0 = other.m_y0;
104 m_y0_destination = std::move(other.m_y0_destination);
105
106 m_points = std::move(other.m_points);
107
108 m_y0_cacheUsed = false;
109 return *this;
110 }
112 virtual ~curve() = default;
113
114 void reset() override;
115
121 bool add_point(ossia::curve_segment<Y>&& segment, X abscissa, Y value);
122
126 bool remove_point(X abscissa);
127
131 Y value_at(X abscissa) const;
132
133 ossia::curve_type get_type() const override;
134
139 X get_x0() const;
140
145 Y get_y0() const;
146
150 void set_x0(X value);
151
155 void set_y0(Y value);
156
159 std::optional<destination> get_y0_destination() const;
160
163 void set_y0_destination(const ossia::destination& destination);
164
169 map_type get_points() const;
170
171 static Y convert_to_template_type_value(
172 const ossia::value& value, ossia::destination_index::const_iterator idx);
173
174 void reserve(std::size_t count) { m_points.reserve(count); }
175
176private:
177 mutable X m_x0;
178 mutable Y m_y0;
179 mutable std::optional<ossia::destination> m_y0_destination;
180
181 mutable map_type m_points;
182
183 mutable Y m_y0_cache;
184
185 mutable bool m_y0_cacheUsed = false;
186};
187
188template <typename X, typename Y>
190{
191 m_y0_cacheUsed = false;
192}
193
194template <typename X, typename Y>
195inline bool
196curve<X, Y>::add_point(ossia::curve_segment<Y>&& segment, X abscissa, Y value)
197{
198 m_points.emplace(abscissa, std::make_pair(value, std::move(segment)));
199
200 return true;
201}
202
203template <typename X, typename Y>
204inline bool curve<X, Y>::remove_point(X abscissa)
205{
206 return m_points.erase(abscissa) > 0;
207}
208
209template <typename X, typename Y>
210inline Y curve<X, Y>::value_at(X abscissa) const
211{
212 X lastAbscissa = get_x0();
213 Y lastValue = get_y0();
214
215 auto end = m_points.end();
216 for(auto it = m_points.begin(); it != end; ++it)
217 {
218 if(abscissa > lastAbscissa && abscissa <= it->first)
219 {
220 lastValue = it->second.second(
221 ((double)abscissa - (double)lastAbscissa)
222 / ((double)it->first - (double)lastAbscissa),
223 lastValue, it->second.first);
224 break;
225 }
226 else if(abscissa > it->first)
227 {
228 lastAbscissa = it->first;
229 lastValue = it->second.first;
230 }
231 else
232 break;
233 }
234
235 return lastValue;
236}
237
238template <typename X, typename Y>
240{
241 static constexpr auto type
242 = std::make_pair(curve_segment_type_map<X>(), curve_segment_type_map<Y>());
243 return type;
244}
245
246template <typename X, typename Y>
247inline X curve<X, Y>::get_x0() const
248{
249 return m_x0;
250}
251
252template <typename X, typename Y>
253inline Y curve<X, Y>::get_y0() const
254{
255 if(!m_y0_destination)
256 return m_y0;
257 else
258 {
259 if(m_y0_cacheUsed)
260 return m_y0_cache;
261
262 const destination& dest = *m_y0_destination;
263 m_y0_cacheUsed = true;
264 m_y0_cache = convert_to_template_type_value(
265 dest.address().fetch_value(), dest.index.begin());
266
267 return m_y0_cache;
268 }
269}
270
271template <typename X, typename Y>
273{
274 m_x0 = value;
275}
276
277template <typename X, typename Y>
279{
280 m_y0 = value;
281}
282
283template <typename X, typename Y>
284inline std::optional<destination> curve<X, Y>::get_y0_destination() const
285{
286 return m_y0_destination;
287}
288
289template <typename X, typename Y>
290inline void curve<X, Y>::set_y0_destination(const destination& destination)
291{
292 m_y0_destination = destination;
293}
294
295template <typename X, typename Y>
296inline typename curve<X, Y>::map_type curve<X, Y>::get_points() const
297{
298 return m_points;
299}
300
301template <typename X, typename Y>
303 const value& v, ossia::destination_index::const_iterator idx)
304{
305 using namespace ossia;
306 using namespace std;
307 struct visitor
308 {
309 destination_index::const_iterator index;
310 Y operator()(int32_t i) const { return i; }
311 Y operator()(float f) const { return f; }
312 Y operator()(bool b) const { return b; }
313 Y operator()(char c) const { return c; }
314 Y operator()(vec2f vec) const { return vec[*index]; }
315 Y operator()(vec3f vec) const { return vec[*index]; }
316 Y operator()(vec4f vec) const { return vec[*index]; }
317 Y operator()(const std::vector<ossia::value>& t) const
318 {
319 auto& val = t[*index];
320 return convert_to_template_type_value(val, index + 1);
321 }
322 Y operator()(const value_map_type& t) const
323 {
324#if defined(__cpp_exceptions)
326 "curve_impl::convertToTemplateTypeValue: "
327 "Cannot convert Map to a numeric type");
328#endif
329 return {};
330 }
331
332 Y operator()(impulse) const
333 {
334#if defined(__cpp_exceptions)
336 "curve_impl::convertToTemplateTypeValue: "
337 "Cannot convert Impulse to a numeric type");
338#endif
339 return {};
340 }
341 Y operator()(const std::string& str) const
342 {
343#if defined(__cpp_exceptions)
345 "curve_impl::convertToTemplateTypeValue: "
346 "Cannot convert String to a numeric type");
347#endif
348 return {};
349 }
350 Y operator()() const
351 {
352#if defined(__cpp_exceptions)
354 "curve_impl::convertToTemplateTypeValue: "
355 "No type provided");
356#endif
357 return {};
358 }
359 } vis{idx};
360
361 return v.apply(vis);
362}
363
375class OSSIA_EXPORT constant_curve final : public curve_abstract
376{
377public:
379 : mValue{std::move(v)}
380 {
381 }
382 ~constant_curve() override;
383 constant_curve() = default;
384 constant_curve(const constant_curve&) = delete;
385 constant_curve(constant_curve&&) = delete;
386 constant_curve& operator=(const constant_curve&) = delete;
387 constant_curve& operator=(constant_curve&&) = delete;
388
389 [[nodiscard]] ossia::value value() const { return mValue; }
390
391 [[nodiscard]] curve_type get_type() const override
392 {
393 return std::make_pair(
394 ossia::curve_segment_type::DOUBLE, ossia::curve_segment_type::ANY);
395 }
396
397 void reset() override { }
398
399private:
400 const ossia::value mValue;
401};
402}
The constant_curve class A curve that always return a same single value.
Definition curve.hpp:376
void reset() override
Clear internal data structures of the curve.
Definition curve.hpp:397
curve_type get_type() const override
get the type of both coordinates
Definition curve.hpp:391
The curve_abstract class.
Definition curve_abstract.hpp:42
The curve class.
Definition curve.hpp:59
X get_x0() const
get initial point abscissa
Definition curve.hpp:247
void reset() override
Clear internal data structures of the curve.
Definition curve.hpp:189
bool remove_point(X abscissa)
remove a point from the curve
Definition curve.hpp:204
void set_x0(X value)
set initial point abscissa
Definition curve.hpp:272
bool add_point(ossia::curve_segment< Y > &&segment, X abscissa, Y value)
add a segment to reach a target point to the curve
Definition curve.hpp:196
std::optional< destination > get_y0_destination() const
get initial point ordinate destination
Definition curve.hpp:284
map_type get_points() const
get points of the curve
Definition curve.hpp:296
Y value_at(X abscissa) const
get value at an abscissa
Definition curve.hpp:210
void set_y0(Y value)
set initial point ordinate
Definition curve.hpp:278
ossia::curve_type get_type() const override
get the type of both coordinates
Definition curve.hpp:239
virtual ~curve()=default
destructor
void set_y0_destination(const ossia::destination &destination)
set initial curve ordinate using a Destination
Definition curve.hpp:290
Y get_y0() const
get initial point ordinate
Definition curve.hpp:253
The value class.
Definition value.hpp:173
Definition git_info.h:7
The invalid_value_type_error class.
Definition network/exceptions.hpp:87