Project
Loading...
Searching...
No Matches
FrameworkGUIDevicesGraph.cxx
Go to the documentation of this file.
1// Copyright 2019-2020 CERN and copyright holders of ALICE O2.
2// See https://alice-o2.web.cern.ch/copyright for details of the copyright holders.
3// All rights not expressly granted are reserved.
4//
5// This software is distributed under the terms of the GNU General Public
6// License v3 (GPL Version 3), copied verbatim in the file "COPYING".
7//
8// In applying this license CERN does not waive the privileges and immunities
9// granted to it by virtue of its status as an Intergovernmental Organization
10// or submit itself to any jurisdiction.
13#include "FrameworkGUIState.h"
14#include "PaletteHelpers.h"
18#include "Framework/Logger.h"
20#include "Framework/Logger.h"
21#include "../src/WorkflowHelpers.h"
22#include "DebugGUI/imgui.h"
23#include <DebugGUI/icons_font_awesome.h>
24#include <algorithm>
25#include <cmath>
26#include <vector>
27
28#pragma GCC diagnostic push
29#pragma GCC diagnostic ignored "-Wpedantic"
30static inline ImVec2 operator+(const ImVec2& lhs, const ImVec2& rhs) { return ImVec2(lhs.x + rhs.x, lhs.y + rhs.y); }
31static inline ImVec2 operator-(const ImVec2& lhs, const ImVec2& rhs) { return ImVec2(lhs.x - rhs.x, lhs.y - rhs.y); }
32
33namespace o2::framework::gui
34{
35struct NodeColor {
36 ImVec4 normal;
37 ImVec4 hovered;
38 ImVec4 title;
40};
41
43
45{
46 if (info.active == false) {
47 return NodeColor{
50 }
51 switch (info.streamingState) {
53 return NodeColor{
55 .hovered = PaletteHelpers::YELLOW,
57 .title_hovered = PaletteHelpers::DARK_YELLOW};
59 return NodeColor{
61 .hovered = PaletteHelpers::GREEN,
62 .title = PaletteHelpers::GREEN,
63 .title_hovered = PaletteHelpers::DARK_GREEN};
65 default:
66 return NodeColor{
69 .title = PaletteHelpers::GRAY,
70 .title_hovered = PaletteHelpers::BLACK};
71 }
72}
73
75const static ImColor INPUT_SLOT_COLOR = {150, 150, 150, 150};
76const static ImColor OUTPUT_SLOT_COLOR = {150, 150, 150, 150};
77const static ImColor NODE_LABEL_BACKGROUND_COLOR = {45, 45, 45, 255};
78const static ImColor NODE_LABEL_TEXT_COLOR = {244, 244, 244, 255};
79const static ImVec4& ERROR_MESSAGE_COLOR = PaletteHelpers::RED;
80const static ImVec4& WARNING_MESSAGE_COLOR = PaletteHelpers::YELLOW;
81const static ImColor ARROW_BACKGROUND_COLOR = {100, 100, 0};
82const static ImColor ARROW_HALFGROUND_COLOR = {170, 170, 70};
83const static ImColor ARROW_COLOR = {200, 200, 100};
84const static ImColor ARROW_SELECTED_COLOR = {200, 0, 100};
85const static ImU32 GRID_COLOR = ImColor(200, 200, 200, 40);
86const static ImColor NODE_BORDER_COLOR = {100, 100, 100};
87const static ImColor LEGEND_COLOR = {100, 100, 100};
88
90const static float GRID_SZ = 64.0f;
91const static float ARROW_BACKGROUND_THICKNESS = 1.f;
92const static float ARROW_HALFGROUND_THICKNESS = 2.f;
93const static float ARROW_THICKNESS = 3.f;
94const static float NODE_BORDER_THICKNESS = 4.f;
95const static ImVec4 LEGEND_BACKGROUND_COLOR = {0.125, 0.180, 0.196, 1};
96
97const static ImVec4 SLOT_EMPTY_COLOR = {0.275, 0.275, 0.275, 1.};
98const static ImVec4& SLOT_PENDING_COLOR = PaletteHelpers::RED;
99const static ImVec4& SLOT_DISPATCHED_COLOR = PaletteHelpers::YELLOW;
100const static ImVec4& SLOT_DONE_COLOR = PaletteHelpers::GREEN;
101
103const float NODE_SLOT_RADIUS = 4.0f;
104const ImVec2 NODE_WINDOW_PADDING(8.0f, 8.0f);
105
107void displayGrid(bool show_grid, ImVec2 offset, ImDrawList* draw_list)
108{
109 if (show_grid == false) {
110 return;
111 }
112 ImVec2 win_pos = ImGui::GetCursorScreenPos();
113 ImVec2 canvas_sz = ImGui::GetWindowSize();
114 for (float x = fmodf(offset.x, GRID_SZ); x < canvas_sz.x; x += GRID_SZ) {
115 draw_list->AddLine(ImVec2(x, 0.0f) + win_pos, ImVec2(x, canvas_sz.y) + win_pos, GRID_COLOR);
116 }
117 for (float y = fmodf(offset.y, GRID_SZ); y < canvas_sz.y; y += GRID_SZ) {
118 draw_list->AddLine(ImVec2(0.0f, y) + win_pos, ImVec2(canvas_sz.x, y) + win_pos, GRID_COLOR);
119 }
120}
121
122void displayLegend(bool show_legend, ImVec2 offset, ImDrawList* draw_list)
123{
124 if (show_legend == false) {
125 return;
126 }
127 struct LegendItem {
128 std::string label;
129 const ImVec4& color;
130 };
131 static auto legend = {
132 LegendItem{" Slot empty", SLOT_EMPTY_COLOR},
133 LegendItem{" Slot pending", SLOT_PENDING_COLOR},
134 LegendItem{" Slot dispatched", SLOT_DISPATCHED_COLOR},
135 LegendItem{" Slot done", SLOT_DONE_COLOR},
136 };
137 ImGui::PushStyleColor(ImGuiCol_WindowBg, LEGEND_BACKGROUND_COLOR);
138 ImGui::PushStyleColor(ImGuiCol_TitleBg, LEGEND_BACKGROUND_COLOR);
139 ImGui::PushStyleColor(ImGuiCol_ResizeGrip, 0);
140
141 ImGui::Begin("Legend");
142 ImGui::Dummy(ImVec2(0.0f, 10.0f));
143 for (auto [label, color] : legend) {
144 ImVec2 vMin = ImGui::GetWindowPos() + ImGui::GetCursorPos() + ImVec2(9, 0);
145 ImVec2 vMax = vMin + ImGui::CalcTextSize(" ");
146 ImGui::PushStyleColor(ImGuiCol_ChildBg, color);
147 ImGui::GetWindowDrawList()->AddRectFilled(vMin, vMax, ImColor(color));
148 ImGui::GetWindowDrawList()->AddRect(vMin, vMax, GRID_COLOR);
149 ImGui::Text("%s", label.data());
150 ImGui::PopStyleColor();
151 }
152 ImGui::End();
153 ImGui::PopStyleColor(3);
154}
155
156#define MAX_GROUP_NAME_SIZE 128
157
158// Private helper struct to keep track of node groups
159struct Group {
160 int ID;
163 Group(int id, char const* n, size_t mid)
164 {
165 ID = id;
166 strncpy(name, n, MAX_GROUP_NAME_SIZE);
167 name[MAX_GROUP_NAME_SIZE - 1] = 0;
168 metadataId = mid;
169 }
170};
171
172constexpr int MAX_SLOTS = 512;
173constexpr int MAX_INPUT_VALUE_SIZE = 24;
174
176 size_t value;
177 char buffer[MAX_INPUT_VALUE_SIZE] = "unknown";
178 float textSize = ImGui::CalcTextSize("unknown").x;
179};
180
182 size_t value;
183 char buffer[MAX_INPUT_VALUE_SIZE] = "unknown";
184 float textSize = ImGui::CalcTextSize("unknown").x;
185};
186
187// Private helper struct for the graph model
188struct Node {
189 int ID;
191 char Name[64];
192 ImVec2 Size;
193 float Value;
194 ImVec4 Color;
198
199 Node(int id, int groupID, char const* name, float value, const ImVec4& color, int inputs_count, int outputs_count)
200 {
201 ID = id;
202 GroupID = groupID;
203 strncpy(Name, name, 63);
204 Name[63] = 0;
205 Value = value;
206 Color = color;
207 InputsCount = inputs_count;
208 OutputsCount = outputs_count;
209 }
210};
211
212// Private helper struct for the layout of the graph
213struct NodePos {
214 ImVec2 pos;
215 static ImVec2 GetInputSlotPos(ImVector<Node> const& infos, ImVector<NodePos> const& positions, int nodeId, int slot_no)
216 {
217 ImVec2 const& pos = positions[nodeId].pos;
218 ImVec2 const& size = infos[nodeId].Size;
219 float inputsCount = infos[nodeId].InputsCount;
220 return ImVec2(pos.x, pos.y + size.y * ((float)slot_no + 1) / (inputsCount + 1));
221 }
222 static ImVec2 GetOutputSlotPos(ImVector<Node> const& infos, ImVector<NodePos> const& positions, int nodeId, int slot_no)
223 {
224 ImVec2 const& pos = positions[nodeId].pos;
225 ImVec2 const& size = infos[nodeId].Size;
226 float outputsCount = infos[nodeId].OutputsCount;
227 return ImVec2(pos.x + size.x, pos.y + size.y * ((float)slot_no + 1) / (outputsCount + 1));
228 }
229};
230
231// Private helper struct for the edges in the graph
232struct NodeLink {
234
235 NodeLink(int input_idx, int input_slot, int output_idx, int output_slot)
236 {
237 InputIdx = input_idx;
238 InputSlot = input_slot;
239 OutputIdx = output_idx;
240 OutputSlot = output_slot;
241 }
242};
243
245template <typename RECORD, typename ITEM, typename CONTEXT>
247 using NumRecordsCallback = std::function<size_t(void)>;
248 using RecordCallback = std::function<RECORD(size_t)>;
249 using NumItemsCallback = std::function<size_t(RECORD const&)>;
250 using ItemCallback = std::function<ITEM const&(RECORD const&, size_t)>;
251 using ValueCallback = std::function<int(ITEM const&)>;
252 using ColorCallback = std::function<ImU32(int value)>;
253 using ContextCallback = std::function<CONTEXT&()>;
254 using PaintCallback = std::function<void(int row, int column, int value, ImU32 color, CONTEXT const& context)>;
255
256 static void draw(const char* name,
257 ImVec2 const& offset,
258 ImVec2 const& sizeHint,
259 CONTEXT const& context,
260 NumRecordsCallback const& getNumRecords,
261 RecordCallback const& getRecord,
262 NumItemsCallback const& getNumItems,
263 ItemCallback const& getItem,
264 ValueCallback const& getValue,
265 ColorCallback const& getColor,
266 PaintCallback const& describeCell)
267 {
268 for (size_t ri = 0; ri < getNumRecords(); ++ri) {
269 auto record = getRecord(ri);
270 for (size_t ii = 0; ii < getNumItems(record); ++ii) {
271 auto item = getItem(record, ii);
272 int value = getValue(item);
273 ImU32 color = getColor(value);
274 ImVec2 pos = ImGui::GetCursorScreenPos();
275 describeCell(ri, ii, value, color, context);
276 }
277 }
278 }
279
281 {
282 return []() -> size_t { return 1UL; };
283 }
284
286 {
287 return [&viewIndex]() -> size_t {
288 if (viewIndex.isComplete()) {
289 return viewIndex.w;
290 }
291 return 0;
292 };
293 }
294
296 {
297 return [](RECORD const& record) -> size_t { return 1UL; };
298 }
299
301 {
302 return [&viewIndex](int record) -> int {
303 if (viewIndex.isComplete()) {
304 return viewIndex.h;
305 }
306 return 0;
307 };
308 }
309
310 template <typename T>
312 Metric2DViewIndex const& viewIndex)
313 {
314 return [&metrics, &viewIndex](RECORD const& record, size_t i) -> ITEM const& {
315 // Calculate the index in the viewIndex.
316 auto idx = record * viewIndex.h + i;
317 assert(viewIndex.indexes.size() > idx);
318 MetricInfo const& metricInfo = metrics.metrics[viewIndex.indexes[idx]];
319 auto& data = DeviceMetricsInfoHelpers::get<T, metricStorageSize<T>()>(metrics, metricInfo.storeIdx);
320 return data[(metricInfo.pos - 1) % data.size()];
321 };
322 }
323
324 template <typename T>
326 {
327 return [](size_t i) -> int {
328 return i;
329 };
330 }
331
332 template <typename T>
334 {
335 return [](ITEM const& item) -> T { return item; };
336 }
337
338 static ColorCallback colorPalette(std::vector<ImU32> const& colors, int minValue, int maxValue)
339 {
340 return [colors, minValue, maxValue](int const& value) -> ImU32 {
341 if (value < minValue) {
342 return colors[0];
343 } else if (value > maxValue) {
344 return colors[colors.size() - 1];
345 } else {
346 int idx = (value - minValue) * (colors.size() - 1) / (maxValue - minValue);
347 return colors[idx];
348 }
349 };
350 }
351};
352
355 ImVector<Node>* nodes;
357 ImVector<NodePos>* positions;
358 ImDrawList* draw_list;
359 ImVec2 offset;
360};
361
363 std::vector<DeviceInfo> const& infos,
364 std::vector<DeviceSpec> const& specs,
365 std::vector<DataProcessingStates> const& allStates,
366 std::vector<DataProcessorInfo> const& metadata,
367 std::vector<DeviceControl>& controls,
368 std::vector<DeviceMetricsInfo> const& metricsInfos)
369{
370 ImGui::SetNextWindowPos(ImVec2(0, 0), 0);
371 if (state.bottomPaneVisible) {
372 ImGui::SetNextWindowSize(ImVec2(ImGui::GetIO().DisplaySize.x, ImGui::GetIO().DisplaySize.y - state.bottomPaneSize), 0);
373 } else {
374 ImGui::SetNextWindowSize(ImVec2(ImGui::GetIO().DisplaySize.x, ImGui::GetIO().DisplaySize.y), 0);
375 }
376
377 ImGui::Begin("Physical topology view", nullptr, ImGuiWindowFlags_NoBringToFrontOnFocus | ImGuiWindowFlags_NoTitleBar | ImGuiWindowFlags_NoCollapse | ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoResize);
378
379 static ImVector<Node> nodes;
380 static ImVector<Group> groups;
381 static ImVector<NodeLink> links;
382 static ImVector<NodePos> positions;
383
384 static bool inited = false;
385 static ImVec2 scrolling = ImVec2(0.0f, 0.0f);
386 static bool show_grid = true;
387 static bool show_legend = true;
388 static int node_selected = -1;
389
390 auto prepareChannelView = [&specs, &metricsInfos, &metadata](ImVector<Node>& nodeList, ImVector<Group>& groupList) {
391 struct LinkInfo {
392 int specId;
393 int outputId;
394 };
395 std::map<std::string, LinkInfo> linkToIndex;
396 for (int si = 0; si < specs.size(); ++si) {
397 int oi = 0;
398 for (auto&& output : specs[si].outputChannels) {
399 linkToIndex.insert(std::make_pair(output.name, LinkInfo{si, oi}));
400 oi += 1;
401 }
402 }
403 // Prepare the list of groups.
404 std::string workflow = "Ungrouped";
405 int groupId = 0;
406 for (size_t mi = 0; mi < metadata.size(); ++mi) {
407 auto const& metadatum = metadata[mi];
408 if (metadatum.executable == workflow) {
409 continue;
410 }
411 workflow = metadatum.executable;
412 char* groupBasename = strrchr(workflow.data(), '/');
413 char const* groupName = groupBasename ? groupBasename + 1 : workflow.data();
414 bool hasDuplicate = false;
415 for (size_t gi = 0; gi < groupList.Size; ++gi) {
416 if (strncmp(groupName, groupList[gi].name, MAX_GROUP_NAME_SIZE - 1) == 0) {
417 hasDuplicate = true;
418 break;
419 }
420 }
421 if (hasDuplicate == false) {
422 groupList.push_back(Group(groupId++, groupName, mi));
423 }
424 }
425 // Do matching between inputs and outputs
426 for (int si = 0; si < specs.size(); ++si) {
427 auto& spec = specs[si];
428 int groupId = 0;
429
430 auto metadatum = std::find_if(metadata.begin(), metadata.end(),
431 [&name = spec.name](DataProcessorInfo const& info) { return info.name == name; });
432
433 for (size_t gi = 0; gi < groupList.Size; ++gi) {
434 if (metadatum == metadata.end()) {
435 break;
436 }
437 const char* groupName = strrchr(metadatum->executable.data(), '/');
438 if (strncmp(groupList[gi].name, groupName ? groupName + 1 : metadatum->executable.data(), 127) == 0) {
439 groupId = gi;
440 break;
441 }
442 }
443 nodeList.push_back(Node(si, groupId, spec.id.c_str(), 0.5f,
444 ImColor(255, 100, 100),
445 spec.inputChannels.size(),
446 spec.outputChannels.size()));
447 int ii = 0;
448 for (auto& input : spec.inputChannels) {
449 auto const& outName = input.name;
450 auto const& out = linkToIndex.find(input.name);
451 if (out == linkToIndex.end()) {
452 LOG(error) << "Could not find suitable node for " << outName;
453 continue;
454 }
455 links.push_back(NodeLink{out->second.specId, out->second.outputId, si, ii});
456 ii += 1;
457 }
458 }
459
460 // ImVector does boudary checks, so I bypass the case there is no
461 // edges.
462 std::vector<TopoIndexInfo> sortedNodes = {{0, 0}};
463 if (links.size()) {
464 sortedNodes = WorkflowHelpers::topologicalSort(specs.size(), &(links[0].InputIdx), &(links[0].OutputIdx), sizeof(links[0]), links.size());
465 }
466 // This is to protect for the cases in which there is a loop in the
467 // definition of the inputs and of the outputs due to the
468 // way the forwarding creates hidden dependencies between processes.
469 // This should not happen, but apparently it does.
470 for (auto di = 0; di < specs.size(); ++di) {
471 auto fn = std::find_if(sortedNodes.begin(), sortedNodes.end(), [di](TopoIndexInfo const& info) {
472 return di == info.index;
473 });
474 if (fn == sortedNodes.end()) {
475 sortedNodes.push_back({(int)di, 0});
476 }
477 }
478 assert(specs.size() == sortedNodes.size());
480 std::sort(sortedNodes.begin(), sortedNodes.end());
481
482 std::vector<int> layerEntries(1024, 0);
483 std::vector<int> layerMax(1024, 0);
484 for (auto& node : sortedNodes) {
485 layerMax[node.layer < 1023 ? node.layer : 1023] += 1;
486 }
487
488 // FIXME: display nodes using topological sort
489 // Update positions
490 for (int si = 0; si < specs.size(); ++si) {
491 auto& node = sortedNodes[si];
492 assert(node.index == si);
493 int xpos = 40 + 240 * node.layer;
494 int ypos = 300 + (std::max(600, 60 * (int)layerMax[node.layer]) / (layerMax[node.layer] + 1)) * (layerEntries[node.layer] - layerMax[node.layer] / 2);
495 positions.push_back(NodePos{ImVec2(xpos, ypos)});
496 layerEntries[node.layer] += 1;
497 }
498 };
499
500 if (!inited) {
501 prepareChannelView(nodes, groups);
502 inited = true;
503 }
504
505 // Create our child canvas
506 ImGui::BeginGroup();
507 ImGui::Checkbox("Show grid", &show_grid);
508 ImGui::SameLine();
509 ImGui::Checkbox("Show legend", &show_legend);
510 ImGui::SameLine();
511 if (ImGui::Button("Center")) {
512 scrolling = ImVec2(0., 0.);
513 }
514 ImGui::SameLine();
515 if (state.leftPaneVisible == false && ImGui::Button("Show tree")) {
516 state.leftPaneVisible = true;
517 }
518 if (state.leftPaneVisible == true && ImGui::Button("Hide tree")) {
519 state.leftPaneVisible = false;
520 }
521 ImGui::SameLine();
522 if (state.bottomPaneVisible == false && ImGui::Button("Show metrics")) {
523 state.bottomPaneVisible = true;
524 }
525 if (state.bottomPaneVisible == true && ImGui::Button("Hide metrics")) {
526 state.bottomPaneVisible = false;
527 }
528 ImGui::SameLine();
529 if (state.rightPaneVisible == false && ImGui::Button("Show inspector")) {
530 state.rightPaneVisible = true;
531 }
532 if (state.rightPaneVisible == true && ImGui::Button("Hide inspector")) {
533 state.rightPaneVisible = false;
534 }
535 ImGui::Separator();
536 ImGui::EndGroup();
537 auto toolbarSize = ImGui::GetItemRectSize();
538 // Draw a list of nodes on the left side
539 bool open_context_menu = false;
540 int node_hovered_in_list = -1;
541 int node_hovered_in_scene = -1;
542 if (state.leftPaneVisible) {
543 ImGui::BeginChild("node_list", ImVec2(state.leftPaneSize, 0));
544 ImGui::Text("Workflows %d", groups.Size);
545 ImGui::Separator();
546 for (int groupId = 0; groupId < groups.Size; groupId++) {
547 Group* group = &groups[groupId];
548 if (ImGui::TreeNodeEx(group->name, ImGuiTreeNodeFlags_DefaultOpen)) {
549 for (int node_idx = 0; node_idx < nodes.Size; node_idx++) {
550 Node* node = &nodes[node_idx];
551 if (node->GroupID != groupId) {
552 continue;
553 }
554 ImGui::Indent(15);
555 ImGui::PushID(node->ID);
556 if (ImGui::Selectable(node->Name, node->ID == node_selected)) {
557 if (ImGui::IsMouseDoubleClicked(0)) {
558 controls[node_selected].logVisible = true;
559 }
560 node_selected = node->ID;
561 }
562 if (ImGui::IsItemHovered()) {
563 node_hovered_in_list = node->ID;
564 open_context_menu |= ImGui::IsMouseClicked(1);
565 }
566 ImGui::PopID();
567 ImGui::Unindent(15);
568 }
569 ImGui::TreePop();
570 }
571 }
572 ImGui::EndChild();
573 ImGui::SameLine();
574 }
575
576 ImGui::BeginGroup();
577
578 ImGui::PushStyleVar(ImGuiStyleVar_FramePadding, ImVec2(1, 1));
579 ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding, ImVec2(0, 0));
580#if defined(ImGuiCol_ChildWindowBg)
581 ImGui::PushStyleColor(ImGuiCol_ChildWindowBg, (ImU32)ImColor(60, 60, 70, 200));
582#else
583 ImGui::PushStyleColor(ImGuiCol_WindowBg, (ImU32)ImColor(60, 60, 70, 200));
584#endif
585 ImVec2 graphSize = ImGui::GetWindowSize();
586 if (state.leftPaneVisible) {
587 graphSize.x -= state.leftPaneSize;
588 }
589 if (state.rightPaneVisible) {
590 graphSize.x -= state.rightPaneSize;
591 }
592 graphSize.y -= toolbarSize.y + 20;
593 ImGui::BeginChild("scrolling_region", graphSize, true, ImGuiWindowFlags_NoScrollbar | ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoScrollWithMouse);
594 ImGui::PushItemWidth(graphSize.x);
595
596 ImVec2 offset = ImGui::GetCursorScreenPos() - scrolling;
597 ImDrawList* draw_list = ImGui::GetWindowDrawList();
598 // Number of layers we need. 2 per node, plus 2 for
599 // the background stuff.
600 draw_list->ChannelsSplit((nodes.Size + 2) * 2);
601
602 // Display grid
603 displayGrid(show_grid, offset, draw_list);
604
605 ImVec2 win_pos = ImGui::GetCursorScreenPos();
606 ImVec2 canvas_sz = ImGui::GetWindowSize();
607 // Display links but only if they are inside the view.
608 for (int link_idx = 0; link_idx < links.Size; link_idx++) {
609 // Do the geometry culling upfront.
610 NodeLink const& link = links[link_idx];
611 ImVec2 p1 = offset + NodePos::GetOutputSlotPos(nodes, positions, link.InputIdx, link.InputSlot);
612 ImVec2 p2 = ImVec2(-3 * NODE_SLOT_RADIUS, 0) + offset + NodePos::GetInputSlotPos(nodes, positions, link.OutputIdx, link.OutputSlot);
613
614 if ((p1.x > win_pos.x + canvas_sz.x + 50) && (p2.x > win_pos.x + canvas_sz.x + 50)) {
615 continue;
616 }
617
618 if ((p1.y > win_pos.y + canvas_sz.y + 50) && (p2.y > win_pos.y + canvas_sz.y + 50)) {
619 continue;
620 }
621
622 if ((p1.y < win_pos.y) && (p2.y < win_pos.y)) {
623 continue;
624 }
625
626 if ((p1.x < win_pos.x) && (p2.x < win_pos.x)) {
627 continue;
628 }
629 draw_list->ChannelsSetCurrent(0); // Background
630 auto color = ARROW_BACKGROUND_COLOR;
631 auto thickness = ARROW_BACKGROUND_THICKNESS;
632
633 bool p1Inside = false;
634 bool p2Inside = false;
635 if ((p1.x > win_pos.x) && (p1.x < (win_pos.x + canvas_sz.x)) && (p1.y < (win_pos.y + canvas_sz.y)) && (p1.y > win_pos.y)) {
636 p1Inside = true;
637 }
638
639 if ((p2.x > win_pos.x) && (p2.x < (win_pos.x + canvas_sz.x)) && (p2.y < (win_pos.y + canvas_sz.y)) && (p2.y > win_pos.y)) {
640 p2Inside = true;
641 }
642
643 if (p1Inside && p2Inside) {
644 // Whatever the two edges completely within the view, gets brighter color and foreground.
645 draw_list->ChannelsSetCurrent(2);
646 color = ARROW_COLOR;
647 thickness = ARROW_THICKNESS;
648 } else if (p1Inside || p2Inside) {
649 draw_list->ChannelsSetCurrent(1);
650 // Whenever one of the two ends is within the view, increase the color but keep the background
651 color = ARROW_HALFGROUND_COLOR;
652 thickness = ARROW_HALFGROUND_THICKNESS;
653 }
654
655 // If something belongs to a selected node, it gets the selected color.
656 if (link.InputIdx == node_selected || link.OutputIdx == node_selected) {
657 auto foregroundLayer = (nodes.Size + 1) * 2 + 1;
658 draw_list->ChannelsSetCurrent(foregroundLayer);
659 color = ARROW_SELECTED_COLOR;
660 thickness = thickness + 2;
661 }
662
663 draw_list->AddBezierCurve(p1, p1 + ImVec2(+50, 0), p2 + ImVec2(-50, 0), p2, color, thickness);
664 }
665
666 auto fgDrawList = ImGui::GetForegroundDrawList();
667 // Display nodes
668 for (int node_idx = 0; node_idx < nodes.Size; node_idx++) {
669 auto backgroundLayer = (node_idx + 1) * 2;
670 auto foregroundLayer = (node_idx + 1) * 2 + 1;
671 // Selected node goes to front
672 if (node_selected == node_idx) {
673 backgroundLayer = (nodes.Size + 1) * 2;
674 foregroundLayer = (nodes.Size + 1) * 2 + 1;
675 }
676 Node* node = &nodes[node_idx];
677 NodePos* pos = &positions[node_idx];
678 const DeviceInfo& info = infos[node_idx];
679
680 ImVec2 node_rect_min = offset + pos->pos;
681
682 // Do not even start if we are sure the box is not visible
683 if ((node_rect_min.x > ImGui::GetCursorScreenPos().x + ImGui::GetWindowSize().x + 50) ||
684 (node_rect_min.y > ImGui::GetCursorScreenPos().y + ImGui::GetWindowSize().y + 50)) {
685 continue;
686 }
687
688 ImGui::PushID(node->ID);
689
690 // Display node contents first
691 draw_list->ChannelsSetCurrent(foregroundLayer);
692 bool old_any_active = ImGui::IsAnyItemActive();
693 ImGui::SetCursorScreenPos(node_rect_min + NODE_WINDOW_PADDING);
694 ImGui::BeginGroup(); // Lock horizontal position
695 ImGui::TextUnformatted(node->Name);
696 switch (info.maxLogLevel) {
697 case LogLevel::Critical:
698 ImGui::SameLine();
699 ImGui::TextColored(ERROR_MESSAGE_COLOR, "%s", ICON_FA_EXCLAMATION_CIRCLE);
700 break;
701 case LogLevel::Error:
702 ImGui::SameLine();
703 ImGui::TextColored(ERROR_MESSAGE_COLOR, "%s", ICON_FA_EXCLAMATION_CIRCLE);
704 break;
705 case LogLevel::Alarm:
706 ImGui::SameLine();
707 ImGui::TextColored(WARNING_MESSAGE_COLOR, "%s", ICON_FA_EXCLAMATION_TRIANGLE);
708 break;
709 case LogLevel::Warning:
710 ImGui::SameLine();
711 ImGui::TextColored(WARNING_MESSAGE_COLOR, "%s", ICON_FA_EXCLAMATION_TRIANGLE);
712 break;
713 default:
714 break;
715 }
716
717 gui::displayDataRelayer(metricsInfos[node->ID], infos[node->ID], specs[node->ID], allStates[node->ID], ImVec2(200., 160.), controls[node->ID].firstWnd);
718 ImGui::EndGroup();
719
720 // Save the size of what we have emitted and whether any of the widgets are being used
721 bool node_widgets_active = (!old_any_active && ImGui::IsAnyItemActive());
722 float attemptX = std::max(ImGui::GetItemRectSize().x, 150.f);
723 float attemptY = std::min(ImGui::GetItemRectSize().y, 128.f);
724 node->Size = ImVec2(attemptX, attemptY) + NODE_WINDOW_PADDING + NODE_WINDOW_PADDING;
725 ImVec2 node_rect_max = node_rect_min + node->Size;
726 ImVec2 node_rect_title = node_rect_min + ImVec2(node->Size.x, 24);
727
728 if (node_rect_min.x > 20 + 2 * NODE_WINDOW_PADDING.x + state.leftPaneSize + graphSize.x) {
729 ImGui::PopID();
730 continue;
731 }
732 if (node_rect_min.y > 20 + 2 * NODE_WINDOW_PADDING.y + toolbarSize.y + graphSize.y) {
733 ImGui::PopID();
734 continue;
735 }
736
737 // Display node box
738 draw_list->ChannelsSetCurrent(backgroundLayer); // Background
739 ImGui::SetCursorScreenPos(node_rect_min);
740 ImGui::InvisibleButton("node", node->Size);
741 if (ImGui::IsItemHovered()) {
742 node_hovered_in_scene = node->ID;
743 open_context_menu |= ImGui::IsMouseClicked(1);
744 if (ImGui::IsMouseDoubleClicked(0)) {
745 controls[node->ID].logVisible = true;
746 }
747 }
748 bool node_moving_active = ImGui::IsItemActive();
749 if (node_widgets_active || node_moving_active) {
750 node_selected = node->ID;
751 }
752 if (node_moving_active && ImGui::IsMouseDragging(0)) {
753 pos->pos = pos->pos + ImGui::GetIO().MouseDelta;
754 }
755 if (ImGui::IsWindowHovered() && !node_moving_active && ImGui::IsMouseDragging(0)) {
756 scrolling = scrolling - ImVec2(ImGui::GetIO().MouseDelta.x / 4.f, ImGui::GetIO().MouseDelta.y / 4.f);
757 }
758
759 auto nodeBg = decideColorForNode(info);
760
761 auto hovered = (node_hovered_in_list == node->ID || node_hovered_in_scene == node->ID || (node_hovered_in_list == -1 && node_selected == node->ID));
762 ImVec4 nodeBgColor = hovered ? nodeBg.hovered : nodeBg.normal;
763 ImVec4 nodeTitleColor = hovered ? nodeBg.title_hovered : nodeBg.title;
764 ImU32 node_bg_color = ImGui::ColorConvertFloat4ToU32(nodeBgColor);
765 ImU32 node_title_color = ImGui::ColorConvertFloat4ToU32(nodeTitleColor);
766
767 draw_list->AddRectFilled(node_rect_min + ImVec2(3.f, 3.f), node_rect_max + ImVec2(3.f, 3.f), ImColor(0, 0, 0, 70), 4.0f);
768 draw_list->AddRectFilled(node_rect_min, node_rect_max, node_bg_color, 4.0f);
769 draw_list->AddRectFilled(node_rect_min, node_rect_title, node_title_color, 4.0f);
770 draw_list->AddRect(node_rect_min, node_rect_max, NODE_BORDER_COLOR, NODE_BORDER_THICKNESS);
771
772 for (int slot_idx = 0; slot_idx < node->InputsCount; slot_idx++) {
773 draw_list->ChannelsSetCurrent(backgroundLayer); // Background
775 auto slotPos = NodePos::GetInputSlotPos(nodes, positions, node_idx, slot_idx);
776 auto pp1 = p1 + offset + slotPos;
777 auto pp2 = p2 + offset + slotPos;
778 auto pp3 = p3 + offset + slotPos;
779 auto color = ARROW_COLOR;
780 if (node_idx == node_selected) {
781 color = ARROW_SELECTED_COLOR;
782 }
783 draw_list->AddTriangleFilled(pp1, pp2, pp3, color);
784 draw_list->AddCircleFilled(offset + slotPos, NODE_SLOT_RADIUS, INPUT_SLOT_COLOR);
785 }
786
787 draw_list->ChannelsSetCurrent(foregroundLayer);
788 MetricLabelsContext context{&nodes, node_idx, &positions, draw_list, offset};
791 "input_labels",
792 offset,
793 ImVec2{node->Size.x, node->Size.y},
794 context,
800 MetricsPainter<int, uint64_t, MetricLabelsContext>::colorPalette(std::vector<ImU32>{{ImColor(0, 100, 0, 255), ImColor(0, 0, 100, 255), ImColor(100, 0, 0, 255)}}, 0, 3),
801 [](int, int item, int value, ImU32 color, MetricLabelsContext const& context) {
802 auto draw_list = context.draw_list;
803 auto offset = context.offset;
804 auto slotPos = NodePos::GetInputSlotPos(nodes, positions, context.nodeIdx, item);
805 Node* node = &nodes[context.nodeIdx];
806 auto& label = node->oldestPossibleInput[item];
807 // Avoid recomputing if the value is the same.
808 if (label.value != value) {
809 label.value = value;
810 snprintf(label.buffer, sizeof(label.buffer), "%d", value);
811 label.textSize = ImGui::CalcTextSize(label.buffer).x;
812 }
813 draw_list->AddRectFilled(offset + slotPos - ImVec2{node->oldestPossibleInput[item].textSize + 5.f * NODE_SLOT_RADIUS, 2 * NODE_SLOT_RADIUS},
814 offset + slotPos + ImVec2{-4.5f * NODE_SLOT_RADIUS, 2 * NODE_SLOT_RADIUS}, NODE_LABEL_BACKGROUND_COLOR, 2., ImDrawFlags_RoundCornersAll);
815 draw_list->AddText(nullptr, 12,
816 offset + slotPos - ImVec2{node->oldestPossibleInput[item].textSize + 4.5f * NODE_SLOT_RADIUS, 2 * NODE_SLOT_RADIUS},
817 NODE_LABEL_TEXT_COLOR,
818 node->oldestPossibleInput[item].buffer);
819 });
820
821 for (int slot_idx = 0; slot_idx < node->OutputsCount; slot_idx++) {
822 draw_list->AddCircleFilled(offset + NodePos::GetOutputSlotPos(nodes, positions, node_idx, slot_idx), NODE_SLOT_RADIUS, OUTPUT_SLOT_COLOR);
823 }
824
826 "output_labels",
827 offset,
828 ImVec2{node->Size.x, node->Size.y},
829 context,
835 MetricsPainter<int, uint64_t, MetricLabelsContext>::colorPalette(std::vector<ImU32>{{ImColor(0, 100, 0, 255), ImColor(0, 0, 100, 255), ImColor(100, 0, 0, 255)}}, 0, 3),
836 [](int, int item, int value, ImU32 color, MetricLabelsContext const& context) {
837 auto draw_list = context.draw_list;
838 auto offset = context.offset;
839 auto slotPos = NodePos::GetOutputSlotPos(nodes, positions, context.nodeIdx, item);
840 Node* node = &nodes[context.nodeIdx];
841 auto& label = node->oldestPossibleOutput[item];
842 // Avoid recomputing if the value is the same.
843 if (label.value != value) {
844 label.value = value;
845 snprintf(label.buffer, sizeof(label.buffer), "%d", value);
846 label.textSize = ImGui::CalcTextSize(label.buffer).x;
847 }
848 auto rectTL = ImVec2{4.5f * NODE_SLOT_RADIUS, -2 * NODE_SLOT_RADIUS};
849 auto rectBR = ImVec2{node->oldestPossibleOutput[item].textSize + 5.f * NODE_SLOT_RADIUS, 2 * NODE_SLOT_RADIUS};
850 draw_list->AddRectFilled(offset + slotPos + rectTL,
851 offset + slotPos + rectBR, NODE_LABEL_BACKGROUND_COLOR, 2., ImDrawFlags_RoundCornersAll);
852 draw_list->AddText(nullptr, 12,
853 offset + slotPos + rectTL,
854 NODE_LABEL_TEXT_COLOR,
855 node->oldestPossibleOutput[item].buffer);
856 });
857
858 ImGui::PopID();
859 }
860 draw_list->ChannelsMerge();
861 displayLegend(show_legend, offset, draw_list);
862
863 // Open context menu
864 if (!ImGui::IsAnyItemHovered() && ImGui::IsWindowHovered(ImGuiHoveredFlags_AnyWindow) && ImGui::IsMouseClicked(1)) {
865 node_selected = node_hovered_in_list = node_hovered_in_scene = -1;
866 open_context_menu = true;
867 }
868 if (open_context_menu) {
869 ImGui::OpenPopup("context_menu");
870 if (node_hovered_in_list != -1) {
871 node_selected = node_hovered_in_list;
872 }
873 if (node_hovered_in_scene != -1) {
874 node_selected = node_hovered_in_scene;
875 }
876 }
877
878 // Scrolling
879 // if (ImGui::IsWindowHovered() && !ImGui::IsAnyItemActive() && ImGui::IsMouseDragging(2, 0.0f))
880 // scrolling = scrolling - ImGui::GetIO().MouseDelta;
881
882 ImGui::PopItemWidth();
883 ImGui::EndChild();
884 ImGui::PopStyleColor();
885 ImGui::PopStyleVar(2);
886 ImGui::EndGroup();
887
888 if (state.rightPaneVisible) {
889 ImGui::SameLine();
890 ImGui::BeginGroup();
891 ImGui::BeginChild("inspector");
892 ImGui::TextUnformatted("Device Inspector");
893 ImGui::Separator();
894 if (node_selected != -1) {
895 auto& node = nodes[node_selected];
896 auto& spec = specs[node_selected];
897 auto& states = allStates[node_selected];
898 auto& control = controls[node_selected];
899 auto& info = infos[node_selected];
900 auto& metrics = metricsInfos[node_selected];
901 auto& group = groups[node.GroupID];
902 auto& metadatum = metadata[group.metadataId];
903
904 if (state.rightPaneVisible) {
905 gui::displayDeviceInspector(spec, info, states, metrics, metadatum, control);
906 }
907 } else {
908 ImGui::TextWrapped("Select a node in the topology to display information about it");
909 }
910 ImGui::EndChild();
911 ImGui::EndGroup();
912 }
913 ImGui::End();
914}
915
916} // namespace o2::framework::gui
917#pragma GGC diagnostic pop
benchmark::State & state
o2::monitoring::tags::Value Value
#define MAX_GROUP_NAME_SIZE
int32_t i
constexpr int p2()
constexpr int p1()
constexpr to accelerate the coordinates changing
void output(const std::map< std::string, ChannelStat > &channels)
Definition rawdump.cxx:197
uint16_t pos
Definition RawData.h:3
GLdouble n
Definition glcorearb.h:1982
GLint GLenum GLint x
Definition glcorearb.h:403
GLsizei GLuint * groups
Definition glcorearb.h:3984
GLuint buffer
Definition glcorearb.h:655
GLsizeiptr size
Definition glcorearb.h:659
GLuint color
Definition glcorearb.h:1272
GLuint const GLchar * name
Definition glcorearb.h:781
GLsizei GLenum const void GLuint GLsizei GLfloat * metrics
Definition glcorearb.h:5500
GLsizei const GLfloat * value
Definition glcorearb.h:819
GLboolean * data
Definition glcorearb.h:298
GLintptr offset
Definition glcorearb.h:660
GLboolean GLuint group
Definition glcorearb.h:3991
GLuint GLsizei const GLchar * label
Definition glcorearb.h:2519
typedef void(APIENTRYP PFNGLCULLFACEPROC)(GLenum mode)
GLuint id
Definition glcorearb.h:650
GLuint * states
Definition glcorearb.h:4932
State for the main GUI window.
void showTopologyNodeGraph(WorkspaceGUIState &state, std::vector< DeviceInfo > const &infos, std::vector< DeviceSpec > const &specs, std::vector< DataProcessingStates > const &allStates, std::vector< DataProcessorInfo > const &metadata, std::vector< DeviceControl > &controls, std::vector< DeviceMetricsInfo > const &metricsInfos)
NodeColor decideColorForNode(const DeviceInfo &info)
const float NODE_SLOT_RADIUS
Node size.
void displayLegend(bool show_legend, ImVec2 offset, ImDrawList *draw_list)
void displayGrid(bool show_grid, ImVec2 offset, ImDrawList *draw_list)
Displays a grid.
void displayDeviceInspector(DeviceSpec const &spec, DeviceInfo const &info, DataProcessingStates const &states, DeviceMetricsInfo const &metrics, DataProcessorInfo const &metadata, DeviceControl &control)
Helper to display information about a device.
void displayDataRelayer(DeviceMetricsInfo const &, DeviceInfo const &, DeviceSpec const &spec, DataProcessingStates const &states, ImVec2 const &size, int &v)
View of the DataRelayer metrics for a given DeviceInfo.
const ImVec2 NODE_WINDOW_PADDING(8.0f, 8.0f)
@ EndOfStreaming
End of streaming requested, but not notified.
@ Streaming
Data is being processed.
@ Idle
End of streaming notified.
D const SVectorGPU< T, D > & rhs
Definition SMatrixGPU.h:191
Vertex< T > operator-(const Vertex< T > &a, const Vertex< T > &b)
Definition Vertex.h:98
BinCenterView< AxisIterator > operator+(BinCenterView< AxisIterator > lhs, int n)
double getValue(DPVAL dp)
bool active
Whether the device is active (running) or not.
Definition DeviceInfo.h:65
Metric2DViewIndex inputChannelMetricsViewIndex
Definition DeviceInfo.h:76
LogParsingHelpers::LogLevel maxLogLevel
The maximum log level ever seen by this device.
Definition DeviceInfo.h:47
StreamingState streamingState
The current state of the device, as reported by it.
Definition DeviceInfo.h:69
Metric2DViewIndex outputChannelMetricsViewIndex
Definition DeviceInfo.h:79
LogLevel
Possible log levels for device log entries.
int w
The size in X of the metrics.
std::vector< std::size_t > indexes
The row major list of indices for the metrics which compose the 2D view.
int h
The size in Y of the metrics.
bool isComplete() const
Whether or not the view is ready to be used.
static const ImVec4 SHADED_GREEN
static const ImVec4 DARK_GREEN
static const ImVec4 SHADED_YELLOW
static const ImVec4 SHADED_RED
static const ImVec4 LIGHT_GRAY
static const ImVec4 DARK_YELLOW
Helper struct to keep track of the results of the topological sort.
static std::vector< TopoIndexInfo > topologicalSort(size_t nodeCount, int const *edgeIn, int const *edgeOut, size_t byteStride, size_t edgesCount)
Group(int id, char const *n, size_t mid)
Context to draw the labels with the metrics.
std::function< ImU32(int value)> ColorCallback
std::function< void(int row, int column, int value, ImU32 color, CONTEXT const &context)> PaintCallback
std::function< size_t(void)> NumRecordsCallback
std::function< int(ITEM const &)> ValueCallback
std::function< size_t(RECORD const &)> NumItemsCallback
static ColorCallback colorPalette(std::vector< ImU32 > const &colors, int minValue, int maxValue)
static NumRecordsCallback metric2D(Metric2DViewIndex &viewIndex)
static ItemCallback latestMetric(DeviceMetricsInfo const &metrics, Metric2DViewIndex const &viewIndex)
std::function< RECORD(size_t)> RecordCallback
std::function< ITEM const &(RECORD const &, size_t)> ItemCallback
static NumItemsCallback items2D(Metric2DViewIndex const &viewIndex)
static void draw(const char *name, ImVec2 const &offset, ImVec2 const &sizeHint, CONTEXT const &context, NumRecordsCallback const &getNumRecords, RecordCallback const &getRecord, NumItemsCallback const &getNumItems, ItemCallback const &getItem, ValueCallback const &getValue, ColorCallback const &getColor, PaintCallback const &describeCell)
static ImVec2 GetInputSlotPos(ImVector< Node > const &infos, ImVector< NodePos > const &positions, int nodeId, int slot_no)
static ImVec2 GetOutputSlotPos(ImVector< Node > const &infos, ImVector< NodePos > const &positions, int nodeId, int slot_no)
Node(int id, int groupID, char const *name, float value, const ImVec4 &color, int inputs_count, int outputs_count)
OldestPossibleOutput oldestPossibleOutput[MAX_SLOTS]
OldestPossibleInput oldestPossibleInput[MAX_SLOTS]
LOG(info)<< "Compressed in "<< sw.CpuTime()<< " s"
std::vector< int > row