OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
gradient.hpp
1#pragma once
2#include <ossia/dataflow/graph_node.hpp>
3#include <ossia/dataflow/node_process.hpp>
4#include <ossia/dataflow/port.hpp>
5#include <ossia/detail/flat_map.hpp>
8#include <ossia/network/base/parameter.hpp>
9#include <ossia/network/dataspace/color.hpp>
11
12namespace ossia::nodes
13{
14class gradient final : public ossia::graph_node
15{
16public:
17 using grad_type = ossia::flat_map<double, ossia::hunter_lab>;
18
19 static auto clamp_color(ossia::argb col)
20 {
21 using namespace std;
22 for(size_t i = 0; i < col.dataspace_value.size(); i++)
23 col.dataspace_value[i] = ossia::clamp<float>(col.dataspace_value[i], 0.f, 1.f);
24
25 return col;
26 }
27
28 gradient()
29 {
30 ossia::outlet_ptr vp = new ossia::value_outlet;
31 vp->target<ossia::value_port>()->type = ossia::argb_u{};
32 m_outlets.push_back(std::move(vp));
33 }
34
35 std::string label() const noexcept override { return "gradient"; }
36
37 void set_gradient(grad_type t) { m_data = std::move(t); }
38
39 const ossia::unit_t& get_unit() const noexcept
40 {
41 auto outlet = m_outlets.back();
42 if(auto unit = outlet->target<ossia::value_port>()->type.target<ossia::unit_t>())
43 {
44 return *unit;
45 }
46 else
47 {
48 static const ossia::unit_t default_unit = ossia::rgb_u{};
49 return default_unit;
50 }
51 }
52
53 ossia::value get_color(ossia::argb c)
54 {
55 ossia::value res = clamp_color(c).dataspace_value;
56 return ossia::convert(res, ossia::argb_u{}, get_unit());
57 }
58
59 void handle_before_first(const ossia::token_request& tk, int64_t tick_start, double position)
60 {
61 auto& out = *m_outlets[0]->target<ossia::value_port>();
62 auto beg = m_data.begin();
63
64 if(beg->first >= position)
65 {
66 out.write_value(get_color(ossia::argb{beg->second}), tick_start);
67 }
68 else if(!mustTween)
69 {
70 out.write_value(get_color(ossia::argb{beg->second}), tick_start);
71 }
72 else
73 {
74 if(!tween)
75 {
76 auto addr = m_outlets[0]->address.target<ossia::net::parameter_base*>();
77 if(addr && *addr)
78 {
79 // TODO if the curve is in another unit, we have to convert it to the
80 // correct unit.
81 tween = ossia::argb{ossia::convert<ossia::vec4f>((*addr)->value())};
82 }
83 else
84 {
85 tween = ossia::argb{beg->second};
86 }
87 }
88 out.write_value(
89 ease_color(0., *tween, beg->first, beg->second, position), tick_start);
90 }
91 }
92
93 ossia::time_value process_dur;
94 void run(const ossia::token_request& t, ossia::exec_state_facade e) noexcept override
95 {
96 if(this->process_dur.impl <= 0)
97 return;
98
99 auto& out = *m_outlets[0]->target<ossia::value_port>();
100
101 const auto [tick_start, d] = e.timings(t);
102 const double pos = t.position() * double(t.parent_duration.impl) / double(this->process_dur.impl);
103
104 switch(m_data.size())
105 {
106 case 0:
107 out.write_value(get_color(ossia::argb{0., 0., 0., 0.}), tick_start);
108 return;
109 case 1:
110 handle_before_first(t, tick_start, pos);
111 return;
112 default: {
113 auto it_next = m_data.lower_bound(pos);
114 // Before start
115 if(it_next == m_data.begin())
116 {
117 handle_before_first(t, tick_start, pos);
118 }
119 // past end
120 else if(it_next == m_data.end())
121 {
122 out.write_value(get_color(ossia::argb{m_data.rbegin()->second}), tick_start);
123 }
124 else
125 {
126 auto it_prev = it_next;
127 --it_prev;
128
129 out.write_value(
130 ease_color(
131 it_prev->first, it_prev->second, it_next->first, it_next->second,
132 pos),
133 tick_start);
134 }
135 }
136 }
137 }
138
139 ossia::value ease_color(
140 double prev_pos, ossia::hunter_lab prev, double next_pos, ossia::hunter_lab next,
141 double pos)
142 {
143 // Interpolate in La*b* domain
144 const auto coeff = (pos - prev_pos) / (next_pos - prev_pos);
145
146 ossia::hunter_lab res;
147 ossia::easing::ease e{};
148 res.dataspace_value = ossia::make_vec(
149 e(prev.dataspace_value[0], next.dataspace_value[0], coeff),
150 e(prev.dataspace_value[1], next.dataspace_value[1], coeff),
151 e(prev.dataspace_value[2], next.dataspace_value[2], coeff));
152
153 return get_color(ossia::argb{res});
154 }
155
156public:
157 std::optional<ossia::argb> tween;
158
159private:
160 grad_type m_data;
161
162public:
163 bool mustTween{};
164};
165
166class gradient_process final : public ossia::node_process
167{
168public:
169 using ossia::node_process::node_process;
170 void start() override {
171 static_cast<gradient*>(node.get())->tween = std::nullopt;
172 }
173};
174}
The parameter_base class.
Definition ossia/network/base/parameter.hpp:48
The value class.
Definition value.hpp:173
The time_value class.
Definition ossia/editor/scenario/time_value.hpp:30
Definition dataspace.hpp:24