Loading...
Searching...
No Matches
MultiWindowDevice.hpp
1#pragma once
2#include <Gfx/GfxParameter.hpp>
3#include <Gfx/Graph/MultiWindowNode.hpp>
4#include <Gfx/Graph/Window.hpp>
5#include <Gfx/Settings/Model.hpp>
6
7#include <ossia/network/base/device.hpp>
8#include <ossia/network/base/protocol.hpp>
9#include <ossia/network/generic/generic_node.hpp>
10
11#include <ossia-qt/invoke.hpp>
12
13#include <QGuiApplication>
14namespace Gfx
15{
16static score::gfx::MultiWindowNode* createMultiWindowNode(
17 const std::vector<OutputMapping>& mappings,
18 SwapchainFlag swapFlag = SwapchainFlag::NoFlag,
19 SwapchainFormat swapFormat = SwapchainFormat::SDR)
20{
21 const auto& gfx_settings = score::AppContext().settings<Gfx::Settings::Model>();
22
24 double rate = gfx_settings.getRate();
25 if(rate > 0)
26 conf = {.manualRenderingRate = 1000. / rate, .supportsVSync = false};
27 else
28 conf = {.manualRenderingRate = 1000. / 60., .supportsVSync = false};
29
30 auto node = new score::gfx::MultiWindowNode{conf, mappings};
31 node->setSwapchainFlag(swapFlag);
32 node->setSwapchainFormat(swapFormat);
33 return node;
34}
35
36class multiwindow_device : public ossia::net::device_base
37{
39 gfx_node_base m_root;
40 QObject m_qtContext;
41
42 ossia::net::parameter_base* fps_param{};
43 ossia::net::parameter_base* rendersize_param{};
44
45 struct PerWindowParams
46 {
47 ossia::net::parameter_base* scaled_cursor{};
48 ossia::net::parameter_base* gl_cursor{};
49 ossia::net::parameter_base* abs_cursor{};
50 ossia::net::parameter_base* size_param{};
51 ossia::net::parameter_base* pos_param{};
52 ossia::net::parameter_base* source_pos{};
53 ossia::net::parameter_base* source_size{};
54 ossia::net::parameter_base* key_press_code{};
55 ossia::net::parameter_base* key_press_text{};
56 ossia::net::parameter_base* key_release_code{};
57 ossia::net::parameter_base* key_release_text{};
58
59 std::vector<QMetaObject::Connection> connections;
60 };
61 std::vector<PerWindowParams> m_perWindow;
62
63 void update_viewport(PerWindowParams& pw)
64 {
65 if(!rendersize_param || !pw.abs_cursor)
66 return;
67
68 auto v = rendersize_param->value();
69 if(auto val = v.target<ossia::vec2f>())
70 {
71 if((*val)[0] >= 1.f && (*val)[1] >= 1.f)
72 {
73 auto dom = pw.abs_cursor->get_domain();
74 ossia::set_max(dom, *val);
75 pw.abs_cursor->set_domain(std::move(dom));
76 }
77 else if(pw.size_param)
78 {
79 v = pw.size_param->value();
80 if(auto sz = v.target<ossia::vec2f>())
81 {
82 if((*sz)[0] >= 1.f && (*sz)[1] >= 1.f)
83 {
84 auto dom = pw.abs_cursor->get_domain();
85 ossia::set_max(dom, *sz);
86 pw.abs_cursor->set_domain(std::move(dom));
87 }
88 }
89 }
90 }
91 }
92
93public:
95 const std::vector<OutputMapping>& mappings,
96 std::unique_ptr<gfx_protocol_base> proto, std::string name,
97 SwapchainFlag swapFlag = SwapchainFlag::NoFlag,
98 SwapchainFormat swapFormat = SwapchainFormat::SDR)
99 : ossia::net::device_base{std::move(proto)}
100 , m_node{createMultiWindowNode(mappings, swapFlag, swapFormat)}
101 , m_root{*this, *static_cast<gfx_protocol_base*>(m_protocol.get()), m_node, name}
102 {
103 this->m_capabilities.change_tree = true;
104
105 // Connect window signals once windows are created by createOutput()
106 m_node->onWindowsCreated = [this] { connectWindowSignals(); };
107
108 // Global FPS output parameter
109 {
110 auto fps_node = std::make_unique<ossia::net::generic_node>("fps", *this, m_root);
111 fps_param = fps_node->create_parameter(ossia::val_type::FLOAT);
112 m_node->onFps = [this](float fps) { fps_param->push_value(fps); };
113 m_root.add_child(std::move(fps_node));
114 }
115
116 // Global rendersize parameter
117 {
118 auto rs_node
119 = std::make_unique<ossia::net::generic_node>("rendersize", *this, m_root);
120 ossia::net::set_description(*rs_node, "Set to [0, 0] to use the viewport's size");
121 rendersize_param = rs_node->create_parameter(ossia::val_type::VEC2F);
122 rendersize_param->push_value(ossia::vec2f{0.f, 0.f});
123 rendersize_param->add_callback([this](const ossia::value& v) {
124 if(auto val = v.target<ossia::vec2f>())
125 {
126 m_node->setRenderSize({(int)(*val)[0], (int)(*val)[1]});
127 for(auto& pw : m_perWindow)
128 update_viewport(pw);
129 }
130 });
131 m_root.add_child(std::move(rs_node));
132 }
133
134 // Per-window subtrees: /0, /1, /2, ...
135 m_perWindow.resize(mappings.size());
136
137 for(int i = 0; i < (int)mappings.size(); ++i)
138 {
139 auto& pw = m_perWindow[i];
140 const auto& mapping = mappings[i];
141 const auto idx_str = std::to_string(i);
142
143 auto win_node = std::make_unique<ossia::net::generic_node>(idx_str, *this, m_root);
144
145 // /i/cursor
146 {
147 auto cursor_node
148 = std::make_unique<ossia::net::generic_node>("cursor", *this, *win_node);
149 {
150 auto scale_node = std::make_unique<ossia::net::generic_node>(
151 "scaled", *this, *cursor_node);
152 pw.scaled_cursor = scale_node->create_parameter(ossia::val_type::VEC2F);
153 pw.scaled_cursor->set_domain(ossia::make_domain(0.f, 1.f));
154 pw.scaled_cursor->push_value(ossia::vec2f{0.f, 0.f});
155 cursor_node->add_child(std::move(scale_node));
156 }
157 {
158 auto scale_node
159 = std::make_unique<ossia::net::generic_node>("gl", *this, *cursor_node);
160 pw.gl_cursor = scale_node->create_parameter(ossia::val_type::VEC2F);
161 pw.gl_cursor->set_domain(ossia::make_domain(0.f, 1.f));
162 pw.gl_cursor->push_value(ossia::vec2f{0.f, 0.f});
163 cursor_node->add_child(std::move(scale_node));
164 }
165 {
166 auto abs_node = std::make_unique<ossia::net::generic_node>(
167 "absolute", *this, *cursor_node);
168 pw.abs_cursor = abs_node->create_parameter(ossia::val_type::VEC2F);
169 pw.abs_cursor->set_domain(
170 ossia::make_domain(
171 ossia::vec2f{0.f, 0.f}, ossia::vec2f{
172 (float)mapping.windowSize.width(),
173 (float)mapping.windowSize.height()}));
174 pw.abs_cursor->push_value(ossia::vec2f{0.f, 0.f});
175 cursor_node->add_child(std::move(abs_node));
176 }
177 win_node->add_child(std::move(cursor_node));
178 }
179
180 // /i/size
181 {
182 auto size_node
183 = std::make_unique<ossia::net::generic_node>("size", *this, *win_node);
184 pw.size_param = size_node->create_parameter(ossia::val_type::VEC2F);
185 pw.size_param->push_value(
186 ossia::vec2f{
187 (float)mapping.windowSize.width(), (float)mapping.windowSize.height()});
188 pw.size_param->add_callback([this, i](const ossia::value& v) {
189 if(auto val = v.target<ossia::vec2f>())
190 {
191 ossia::qt::run_async(&m_qtContext, [this, i, v = *val] {
192 const auto& outputs = m_node->windowOutputs();
193 if(i < (int)outputs.size())
194 {
195 if(auto& w = outputs[i].window)
196 w->resize(QSize{(int)v[0], (int)v[1]});
197 }
198 });
199 if(i < (int)m_perWindow.size())
200 update_viewport(m_perWindow[i]);
201 }
202 });
203 win_node->add_child(std::move(size_node));
204 }
205
206 // /i/position
207 {
208 auto pos_node
209 = std::make_unique<ossia::net::generic_node>("position", *this, *win_node);
210 pw.pos_param = pos_node->create_parameter(ossia::val_type::VEC2F);
211 pw.pos_param->push_value(
212 ossia::vec2f{
213 (float)mapping.windowPosition.x(), (float)mapping.windowPosition.y()});
214 pw.pos_param->add_callback([this, i](const ossia::value& v) {
215 if(auto val = v.target<ossia::vec2f>())
216 {
217 ossia::qt::run_async(&m_qtContext, [this, i, v = *val] {
218 const auto& outputs = m_node->windowOutputs();
219 if(i < (int)outputs.size())
220 {
221 if(auto& w = outputs[i].window)
222 w->setPosition(QPoint{(int)v[0], (int)v[1]});
223 }
224 });
225 }
226 });
227 win_node->add_child(std::move(pos_node));
228 }
229
230 // /i/fullscreen
231 {
232 auto fs_node
233 = std::make_unique<ossia::net::generic_node>("fullscreen", *this, *win_node);
234 auto fs_param = fs_node->create_parameter(ossia::val_type::BOOL);
235 fs_param->push_value(mapping.fullscreen);
236 fs_param->add_callback([this, i](const ossia::value& v) {
237 if(auto val = v.target<bool>())
238 {
239 ossia::qt::run_async(&m_qtContext, [this, i, v = *val] {
240 const auto& outputs = m_node->windowOutputs();
241 if(i < (int)outputs.size())
242 {
243 if(auto& w = outputs[i].window)
244 {
245 if(v)
246 w->showFullScreen();
247 else
248 w->showNormal();
249 }
250 }
251 });
252 }
253 });
254 win_node->add_child(std::move(fs_node));
255 }
256
257 // /i/screen (accepts an integer index or a screen name string)
258 {
259 auto screen_node
260 = std::make_unique<ossia::net::generic_node>("screen", *this, *win_node);
261 auto screen_param = screen_node->create_parameter(ossia::val_type::STRING);
262 screen_param->push_value(
263 mapping.screenIndex >= 0 ? std::to_string(mapping.screenIndex)
264 : std::string{});
265 screen_param->add_callback([this, i](const ossia::value& v) {
266 auto apply = [this, i](const std::string& scr) {
267 const auto& outputs = m_node->windowOutputs();
268 if(i >= (int)outputs.size())
269 return;
270 auto& w = outputs[i].window;
271 if(!w)
272 return;
273
274 const auto& cur_screens = qGuiApp->screens();
275
276 // Try parsing as integer index first
277 bool ok = false;
278 int idx = QString::fromStdString(scr).toInt(&ok);
279 if(ok)
280 {
281 if(ossia::valid_index(idx, cur_screens))
282 w->setScreen(cur_screens[idx]);
283 return;
284 }
285
286 // Otherwise match by screen name
287 for(auto s : cur_screens)
288 {
289 if(s->name().toStdString() == scr)
290 {
291 w->setScreen(s);
292 return;
293 }
294 }
295 };
296
297 if(auto val = v.target<std::string>())
298 {
299 ossia::qt::run_async(
300 &m_qtContext, [apply, scr = *val] { apply(scr); });
301 }
302 else if(auto val = v.target<int>())
303 {
304 ossia::qt::run_async(
305 &m_qtContext, [apply, scr = std::to_string(*val)] { apply(scr); });
306 }
307 });
308 win_node->add_child(std::move(screen_node));
309 }
310
311 // /i/source (UV mapping rectangle)
312 {
313 auto source_node
314 = std::make_unique<ossia::net::generic_node>("source", *this, *win_node);
315 {
316 auto pos_node = std::make_unique<ossia::net::generic_node>(
317 "position", *this, *source_node);
318 ossia::net::set_description(*pos_node, "UV position of source rect (0-1)");
319 pw.source_pos = pos_node->create_parameter(ossia::val_type::VEC2F);
320 pw.source_pos->set_domain(ossia::make_domain(0.f, 1.f));
321 pw.source_pos->push_value(
322 ossia::vec2f{
323 (float)mapping.sourceRect.x(), (float)mapping.sourceRect.y()});
324 pw.source_pos->add_callback([this, i](const ossia::value& v) {
325 if(auto val = v.target<ossia::vec2f>())
326 {
327 const auto& outputs = m_node->windowOutputs();
328 if(i < (int)outputs.size())
329 {
330 auto r = outputs[i].sourceRect;
331 r.moveTopLeft(QPointF{(*val)[0], (*val)[1]});
332 m_node->setSourceRect(i, r);
333 }
334 }
335 });
336 source_node->add_child(std::move(pos_node));
337 }
338 {
339 auto sz_node
340 = std::make_unique<ossia::net::generic_node>("size", *this, *source_node);
341 ossia::net::set_description(*sz_node, "UV size of source rect (0-1)");
342 pw.source_size = sz_node->create_parameter(ossia::val_type::VEC2F);
343 pw.source_size->set_domain(ossia::make_domain(0.f, 1.f));
344 pw.source_size->push_value(
345 ossia::vec2f{
346 (float)mapping.sourceRect.width(),
347 (float)mapping.sourceRect.height()});
348 pw.source_size->add_callback([this, i](const ossia::value& v) {
349 if(auto val = v.target<ossia::vec2f>())
350 {
351 const auto& outputs = m_node->windowOutputs();
352 if(i < (int)outputs.size())
353 {
354 auto r = outputs[i].sourceRect;
355 r.setSize(QSizeF{(*val)[0], (*val)[1]});
356 m_node->setSourceRect(i, r);
357 }
358 }
359 });
360 source_node->add_child(std::move(sz_node));
361 }
362 win_node->add_child(std::move(source_node));
363 }
364
365 // /i/blend (soft-edge blending per side)
366 {
367 auto blend_node
368 = std::make_unique<ossia::net::generic_node>("blend", *this, *win_node);
369
370 struct BlendSide
371 {
372 const char* name;
373 int side;
374 float initW;
375 float initG;
376 };
377 BlendSide sides[]
378 = {{"left", 0, mapping.blendLeft.width, mapping.blendLeft.gamma},
379 {"right", 1, mapping.blendRight.width, mapping.blendRight.gamma},
380 {"top", 2, mapping.blendTop.width, mapping.blendTop.gamma},
381 {"bottom", 3, mapping.blendBottom.width, mapping.blendBottom.gamma}};
382
383 for(auto& [sname, side, initW, initG] : sides)
384 {
385 auto side_node
386 = std::make_unique<ossia::net::generic_node>(sname, *this, *blend_node);
387 {
388 auto w_node
389 = std::make_unique<ossia::net::generic_node>("width", *this, *side_node);
390 ossia::net::set_description(*w_node, "Blend width in UV space (0-0.5)");
391 auto w_param = w_node->create_parameter(ossia::val_type::FLOAT);
392 w_param->set_domain(ossia::make_domain(0.f, 0.5f));
393 w_param->push_value(initW);
394 w_param->add_callback([this, i, side](const ossia::value& v) {
395 if(auto val = v.target<float>())
396 {
397 const auto& outputs = m_node->windowOutputs();
398 if(i < (int)outputs.size())
399 {
400 float gamma = 2.2f;
401 switch(side)
402 {
403 case 0:
404 gamma = outputs[i].blendLeft.gamma;
405 break;
406 case 1:
407 gamma = outputs[i].blendRight.gamma;
408 break;
409 case 2:
410 gamma = outputs[i].blendTop.gamma;
411 break;
412 case 3:
413 gamma = outputs[i].blendBottom.gamma;
414 break;
415 }
416 m_node->setEdgeBlend(i, side, *val, gamma);
417 }
418 }
419 });
420 side_node->add_child(std::move(w_node));
421 }
422 {
423 auto g_node
424 = std::make_unique<ossia::net::generic_node>("gamma", *this, *side_node);
425 ossia::net::set_description(*g_node, "Blend curve exponent (0.1-4.0)");
426 auto g_param = g_node->create_parameter(ossia::val_type::FLOAT);
427 g_param->set_domain(ossia::make_domain(0.1f, 4.f));
428 g_param->push_value(initG);
429 g_param->add_callback([this, i, side](const ossia::value& v) {
430 if(auto val = v.target<float>())
431 {
432 const auto& outputs = m_node->windowOutputs();
433 if(i < (int)outputs.size())
434 {
435 float width = 0.f;
436 switch(side)
437 {
438 case 0:
439 width = outputs[i].blendLeft.width;
440 break;
441 case 1:
442 width = outputs[i].blendRight.width;
443 break;
444 case 2:
445 width = outputs[i].blendTop.width;
446 break;
447 case 3:
448 width = outputs[i].blendBottom.width;
449 break;
450 }
451 m_node->setEdgeBlend(i, side, width, *val);
452 }
453 }
454 });
455 side_node->add_child(std::move(g_node));
456 }
457 blend_node->add_child(std::move(side_node));
458 }
459 win_node->add_child(std::move(blend_node));
460 }
461
462 // /i/key
463 {
464 auto key_node
465 = std::make_unique<ossia::net::generic_node>("key", *this, *win_node);
466
467 // /i/key/press
468 {
469 auto press_node
470 = std::make_unique<ossia::net::generic_node>("press", *this, *key_node);
471 {
472 auto code_node
473 = std::make_unique<ossia::net::generic_node>("code", *this, *press_node);
474 pw.key_press_code = code_node->create_parameter(ossia::val_type::INT);
475 press_node->add_child(std::move(code_node));
476 }
477 {
478 auto text_node
479 = std::make_unique<ossia::net::generic_node>("text", *this, *press_node);
480 pw.key_press_text = text_node->create_parameter(ossia::val_type::STRING);
481 press_node->add_child(std::move(text_node));
482 }
483 key_node->add_child(std::move(press_node));
484 }
485
486 // /i/key/release
487 {
488 auto release_node
489 = std::make_unique<ossia::net::generic_node>("release", *this, *key_node);
490 {
491 auto code_node = std::make_unique<ossia::net::generic_node>(
492 "code", *this, *release_node);
493 pw.key_release_code = code_node->create_parameter(ossia::val_type::INT);
494 release_node->add_child(std::move(code_node));
495 }
496 {
497 auto text_node = std::make_unique<ossia::net::generic_node>(
498 "text", *this, *release_node);
499 pw.key_release_text = text_node->create_parameter(ossia::val_type::STRING);
500 release_node->add_child(std::move(text_node));
501 }
502 key_node->add_child(std::move(release_node));
503 }
504
505 win_node->add_child(std::move(key_node));
506 }
507
508 m_root.add_child(std::move(win_node));
509 }
510 }
511
512 // Called after windows are created by createOutput() to connect Qt signals
513 void connectWindowSignals()
514 {
515 const auto& outputs = m_node->windowOutputs();
516 for(int i = 0; i < (int)outputs.size() && i < (int)m_perWindow.size(); ++i)
517 {
518 auto& pw = m_perWindow[i];
519 const auto& wo = outputs[i];
520 if(!wo.window)
521 continue;
522
523 auto* w = wo.window.get();
524
525 // Mouse cursor
526 pw.connections.push_back(
527 QObject::connect(
528 w, &score::gfx::Window::mouseMove,
529 [&pw, w](const QPointF screen, const QPointF win) {
530 auto sz = w->size();
531 if(sz.width() > 0 && sz.height() > 0)
532 {
533 pw.scaled_cursor->push_value(
534 ossia::vec2f{float(win.x() / sz.width()), float(win.y() / sz.height())});
535 pw.gl_cursor->push_value(
536 ossia::vec2f{
537 float(win.x() / sz.width()), 1.f - float(win.y() / sz.height())});
538 pw.abs_cursor->push_value(ossia::vec2f{float(win.x()), float(win.y())});
539 }
540 }));
541
542 // // Window position feedback
543 // pw.connections.push_back(
544 // QObject::connect(w, &QWindow::xChanged, [&pw, w](int x) {
545 // pw.pos_param->set_value(ossia::vec2f{float(x), float(w->y())});
546 // }));
547 // pw.connections.push_back(
548 // QObject::connect(w, &QWindow::yChanged, [&pw, w](int y) {
549 // pw.pos_param->set_value(ossia::vec2f{float(w->x()), float(y)});
550 // }));
551
552 // Keyboard
553 pw.connections.push_back(
554 QObject::connect(w, &score::gfx::Window::key, [&pw](int k, const QString& t) {
555 pw.key_press_code->push_value(k);
556 pw.key_press_text->push_value(t.toStdString());
557 }));
558 pw.connections.push_back(
559 QObject::connect(
560 w, &score::gfx::Window::keyRelease, [&pw](int k, const QString& t) {
561 pw.key_release_code->push_value(k);
562 pw.key_release_text->push_value(t.toStdString());
563 }));
564 }
565 }
566
567 ~multiwindow_device()
568 {
569 // Disconnect all window signals
570 for(auto& pw : m_perWindow)
571 for(auto& c : pw.connections)
572 QObject::disconnect(c);
573 m_perWindow.clear();
574
575 m_node->onWindowsCreated = [] { };
576 m_node->onFps = [](float) { };
577 m_protocol->stop();
578 m_root.clear_children();
579 m_protocol.reset();
580 }
581
582 const gfx_node_base& get_root_node() const override { return m_root; }
583 gfx_node_base& get_root_node() override { return m_root; }
584};
585
586}
Definition score-plugin-gfx/Gfx/Settings/Model.hpp:41
Definition GfxParameter.hpp:60
Definition GfxParameter.hpp:41
Definition MultiWindowDevice.hpp:37
Binds the rendering pipeline to ossia processes.
Definition CameraDevice.cpp:30
STL namespace.
T & settings() const
Access a specific Settings model instance.
Definition ApplicationContext.hpp:41
Definition MultiWindowNode.hpp:11
Definition OutputNode.hpp:61