2#include <Process/Dataflow/Port.hpp>
4#include <score/tools/Debug.hpp>
6#include <ossia/detail/math.hpp>
7#include <ossia/network/domain/domain.hpp>
9#include <QDoubleSpinBox>
14namespace WidgetFactory
18static constexpr auto getMin(
auto& slider)
noexcept
20 if constexpr(
requires { slider.getMin(); })
21 return slider.getMin();
23 return slider.domain().get().template convert_min<T>();
27static constexpr auto getMax(
auto& slider)
noexcept
29 if constexpr(
requires { slider.getMax(); })
30 return slider.getMax();
32 return slider.domain().get().template convert_max<T>();
36static constexpr auto getInit(
auto& slider)
noexcept
38 if constexpr(
requires { slider.getInit(); })
39 return slider.getInit();
40 else if constexpr(
requires { slider.init(); })
41 return ossia::convert<T>(slider.init());
43 return getMin<T>(slider);
48 static constexpr double to01(
double min,
double range,
double val)
noexcept
50 return (val - min) /
range;
53 static constexpr double from01(
double min,
double range,
double val)
noexcept
55 return min + val *
range;
59 static double to01(
const T& slider,
double val)
noexcept
61 auto min = getMin<double>(slider);
62 auto max = getMax<double>(slider);
65 return to01(min, max - min, val);
69 static double from01(
const T& slider,
double val)
noexcept
71 auto min = getMin<double>(slider);
72 auto max = getMax<double>(slider);
75 return from01(min, max - min, val);
79 static std::array<double, 2> to01(
const T& slider, ossia::vec2f val)
noexcept
81 std::array<double, 2> res;
82 const auto min = getMin<ossia::vec2f>(slider);
83 const auto max = getMax<ossia::vec2f>(slider);
84 res[0] = to01(min[0],
double(max[0]) -
double(min[0]), val[0]);
85 res[1] = to01(min[1],
double(max[1]) -
double(min[1]), val[1]);
90 static ossia::vec2f from01(
const T& slider, std::array<double, 2> val)
noexcept
93 const auto min = getMin<ossia::vec2f>(slider);
94 const auto max = getMax<ossia::vec2f>(slider);
95 res[0] = from01(min[0],
double(max[0]) -
double(min[0]), val[0]);
96 res[1] = from01(min[1],
double(max[1]) -
double(min[1]), val[1]);
100 template <
typename T>
101 static std::array<double, 3> to01(
const T& slider, ossia::vec3f val)
noexcept
103 std::array<double, 3> res;
104 const auto min = getMin<ossia::vec3f>(slider);
105 const auto max = getMax<ossia::vec3f>(slider);
106 res[0] = to01(min[0],
double(max[0]) -
double(min[0]), val[0]);
107 res[1] = to01(min[1],
double(max[1]) -
double(min[1]), val[1]);
108 res[2] = to01(min[2],
double(max[2]) -
double(min[2]), val[2]);
112 template <
typename T>
113 static ossia::vec3f from01(
const T& slider, std::array<double, 3> val)
noexcept
116 const auto min = getMin<ossia::vec3f>(slider);
117 const auto max = getMax<ossia::vec3f>(slider);
118 res[0] = from01(min[0],
double(max[0]) -
double(min[0]), val[0]);
119 res[1] = from01(min[1],
double(max[1]) -
double(min[1]), val[1]);
120 res[2] = from01(min[2],
double(max[2]) -
double(min[2]), val[2]);
124 template <
typename T>
125 static std::array<double, 4> to01(
const T& slider, ossia::vec4f val)
noexcept
127 std::array<double, 4> res;
128 const auto min = getMin<ossia::vec4f>(slider);
129 const auto max = getMax<ossia::vec4f>(slider);
130 res[0] = to01(min[0],
double(max[0]) -
double(min[0]), val[0]);
131 res[1] = to01(min[1],
double(max[1]) -
double(min[1]), val[1]);
132 res[2] = to01(min[2],
double(max[2]) -
double(min[2]), val[2]);
133 res[3] = to01(min[3],
double(max[3]) -
double(min[3]), val[3]);
137 template <
typename T>
138 static ossia::vec4f from01(
const T& slider, std::array<double, 4> val)
noexcept
141 const auto min = getMin<ossia::vec4f>(slider);
142 const auto max = getMax<ossia::vec4f>(slider);
143 res[0] = from01(min[0],
double(max[0]) -
double(min[0]), val[0]);
144 res[1] = from01(min[1],
double(max[1]) -
double(min[1]), val[1]);
145 res[2] = from01(min[2],
double(max[2]) -
double(min[2]), val[2]);
146 res[3] = from01(min[3],
double(max[3]) -
double(min[3]), val[3]);
150 template <
typename T, std::
size_t N>
151 static std::array<float, N> from01(
const T& slider, std::array<double, N> val)
noexcept
153 std::array<float, N> res;
154 const auto min = getMin<std::array<double, N>>(slider);
155 const auto max = getMax<std::array<double, N>>(slider);
156 for(std::size_t i = 0; i < N; i++)
157 res[i] = from01(min[i],
double(max[i]) - double(min[i]), val[i]);
164 static double to01(
double min,
double range,
double val)
noexcept
166 return ossia::log_to_normalized(min,
range, val);
169 static double from01(
double min,
double range,
double val)
noexcept
171 return ossia::normalized_to_log(min,
range, val);
174 template <
typename T>
175 static double to01(
const T& slider,
double val)
noexcept
177 auto min = getMin<double>(slider);
178 auto max = getMax<double>(slider);
181 return to01(min, max - min, val);
184 template <
typename T>
185 static double from01(
const T& slider,
double val)
noexcept
187 auto min = getMin<double>(slider);
188 auto max = getMax<double>(slider);
191 return from01(min, max - min, val);
195template <
typename Norm_T>
200 template <
typename T>
203 min = getMin<double>(slider);
204 max = getMax<double>(slider);
209 constexpr double to01(
double val)
const noexcept
211 return Norm_T::to01(min, max - min, val);
214 constexpr double from01(
double val)
const noexcept
216 return Norm_T::from01(min, max - min, val);
220template <
typename Norm_T,
typename Sl
ider_T>
223 const Slider_T& slider;
230 constexpr double to01(
double val)
const noexcept {
return Norm_T::to01(slider, val); }
232 constexpr double from01(
double val)
const noexcept
234 return Norm_T::from01(slider, val);
238template <
typename Control_T,
typename W
idget_T>
239static void initWidgetProperties(Control_T& inlet, Widget_T& widget)
241 if constexpr(
requires { widget.setNoValueChangeOnMove(
true); })
242 widget.setNoValueChangeOnMove(inlet.noValueChangeOnMove);
245template <
typename W
idget_T>
246static void setRange(Widget_T& widget,
auto&& min,
auto&& max,
auto&& init)
248 widget.setRange(min, max, init);
251static inline void setRange(QSpinBox& widget,
auto&& min,
auto&& max,
auto&& init)
253 widget.setRange(min, max);
256static inline void setRange(QDoubleSpinBox& widget,
auto&& min,
auto&& max,
auto&& init)
258 widget.setRange(min, max);
261template <
typename T,
typename Control_T,
typename W
idget_T>
262static void bindFloatDomain(
const T& slider, Control_T& inlet, Widget_T& widget)
264 auto min = getMin<float>(slider);
265 auto max = getMax<float>(slider);
269 setRange(widget, min, max, getInit<float>(slider));
271 if constexpr(std::is_base_of_v<Process::ControlInlet, T>)
273 SCORE_ASSERT(&slider == &inlet);
274 QObject::connect(&inlet, &Control_T::domainChanged, &widget, [&slider, &widget] {
275 auto min = getMin<float>(slider);
276 auto max = getMax<float>(slider);
280 setRange(widget, min, max, getInit<float>(slider));
285template <
typename T,
typename Control_T,
typename W
idget_T>
286static void bindIntDomain(
const T& slider, Control_T& inlet, Widget_T& widget)
288 auto min = getMin<int>(slider);
289 auto max = getMax<int>(slider);
293 setRange(widget, min, max, getInit<int>(slider));
295 if constexpr(std::is_base_of_v<Process::ControlInlet, T>)
297 SCORE_ASSERT(&slider == &inlet);
298 QObject::connect(&inlet, &Control_T::domainChanged, &widget, [&slider, &widget] {
299 auto min = getMin<int>(slider);
300 auto max = getMax<int>(slider);
304 setRange(widget, min, max, getInit<int>(slider));
309template <
typename T,
typename Control_T,
typename W
idget_T>
310static void bindVec2Domain(
const T& slider, Control_T& inlet, Widget_T& widget)
312 auto update_range = [&widget, &inlet, &slider] {
313 auto min = ossia::get_min(inlet.domain());
314 auto max = ossia::get_max(inlet.domain());
315 auto min_float = min.template target<float>();
316 auto max_float = max.template target<float>();
317 if(min_float && max_float)
319 if(*max_float - *min_float == 0)
320 *max_float = *min_float + 1;
322 {*min_float, *min_float}, {*max_float, *max_float}, {*min_float, *min_float});
326 auto min_vec2 = min.template target<ossia::vec2f>();
327 auto max_vec2 = max.template target<ossia::vec2f>();
328 if(min_vec2 && max_vec2)
330 auto& min = *min_vec2;
331 auto& max = *max_vec2;
332 for(std::size_t i = 0; i < min.size(); i++)
334 if(max[i] - min[i] == 0)
338 auto init_vec2 = getInit<ossia::vec2f>(slider);
339 widget.setRange(min, max, init_vec2);
346 if constexpr(std::is_base_of_v<Process::ControlInlet, T>)
348 SCORE_ASSERT(&slider == &inlet);
349 QObject::connect(&inlet, &Control_T::domainChanged, &widget, update_range);
353template <
typename T,
typename Control_T,
typename W
idget_T>
354static void bindVec3Domain(
const T& slider, Control_T& inlet, Widget_T& widget)
356 auto update_range = [&widget, &inlet, &slider] {
357 auto min = ossia::get_min(inlet.domain());
358 auto max = ossia::get_max(inlet.domain());
359 auto min_float = min.template target<float>();
360 auto max_float = max.template target<float>();
361 if(min_float && max_float)
363 if(*max_float - *min_float == 0)
364 *max_float = *min_float + 1;
366 {*min_float, *min_float, *min_float}, {*max_float, *max_float, *max_float});
370 auto min_vec3 = min.template target<ossia::vec3f>();
371 auto max_vec3 = max.template target<ossia::vec3f>();
372 if(min_vec3 && max_vec3)
374 auto& min = *min_vec3;
375 auto& max = *max_vec3;
376 for(std::size_t i = 0; i < min.size(); i++)
378 if(max[i] - min[i] == 0)
382 auto init_vec3 = getInit<ossia::vec3f>(slider);
383 widget.setRange(min, max, init_vec3);
390 if constexpr(std::is_base_of_v<Process::ControlInlet, T>)
392 SCORE_ASSERT(&slider == &inlet);
393 QObject::connect(&inlet, &Control_T::domainChanged, &widget, update_range);
Definition MIDISync.hpp:126