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)
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);
374 ImGui::SetNextWindowSize(ImVec2(ImGui::GetIO().DisplaySize.x, ImGui::GetIO().DisplaySize.y), 0);
377 ImGui::Begin(
"Physical topology view",
nullptr, ImGuiWindowFlags_NoBringToFrontOnFocus | ImGuiWindowFlags_NoTitleBar | ImGuiWindowFlags_NoCollapse | ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoResize);
379 static ImVector<Node>
nodes;
380 static ImVector<Group>
groups;
381 static ImVector<NodeLink> links;
382 static ImVector<NodePos> positions;
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;
390 auto prepareChannelView = [&specs, &metricsInfos, &metadata](ImVector<Node>& nodeList, ImVector<Group>& groupList) {
395 std::map<std::string, LinkInfo> linkToIndex;
396 for (
int si = 0; si < specs.size(); ++si) {
398 for (
auto&&
output : specs[si].outputChannels) {
399 linkToIndex.insert(std::make_pair(
output.name, LinkInfo{si, oi}));
404 std::string workflow =
"Ungrouped";
406 for (
size_t mi = 0; mi < metadata.size(); ++mi) {
407 auto const& metadatum = metadata[mi];
408 if (metadatum.executable == workflow) {
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) {
421 if (hasDuplicate ==
false) {
422 groupList.push_back(
Group(groupId++, groupName, mi));
426 for (
int si = 0; si < specs.size(); ++si) {
427 auto& spec = specs[si];
430 auto metadatum = std::find_if(metadata.begin(), metadata.end(),
433 for (
size_t gi = 0; gi < groupList.Size; ++gi) {
434 if (metadatum == metadata.end()) {
437 const char* groupName = strrchr(metadatum->executable.data(),
'/');
438 if (strncmp(groupList[gi].
name, groupName ? groupName + 1 : metadatum->executable.data(), 127) == 0) {
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()));
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;
455 links.push_back(
NodeLink{out->second.specId, out->second.outputId, si, ii});
462 std::vector<TopoIndexInfo> sortedNodes = {{0, 0}};
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;
474 if (fn == sortedNodes.end()) {
475 sortedNodes.push_back({(
int)
di, 0});
478 assert(specs.size() == sortedNodes.size());
480 std::sort(sortedNodes.begin(), sortedNodes.end());
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;
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;
507 ImGui::Checkbox(
"Show grid", &show_grid);
509 ImGui::Checkbox(
"Show legend", &show_legend);
511 if (ImGui::Button(
"Center")) {
512 scrolling = ImVec2(0., 0.);
515 if (
state.leftPaneVisible ==
false && ImGui::Button(
"Show tree")) {
516 state.leftPaneVisible =
true;
518 if (
state.leftPaneVisible ==
true && ImGui::Button(
"Hide tree")) {
519 state.leftPaneVisible =
false;
522 if (
state.bottomPaneVisible ==
false && ImGui::Button(
"Show metrics")) {
523 state.bottomPaneVisible =
true;
525 if (
state.bottomPaneVisible ==
true && ImGui::Button(
"Hide metrics")) {
526 state.bottomPaneVisible =
false;
529 if (
state.rightPaneVisible ==
false && ImGui::Button(
"Show inspector")) {
530 state.rightPaneVisible =
true;
532 if (
state.rightPaneVisible ==
true && ImGui::Button(
"Hide inspector")) {
533 state.rightPaneVisible =
false;
537 auto toolbarSize = ImGui::GetItemRectSize();
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);
546 for (
int groupId = 0; groupId <
groups.Size; groupId++) {
548 if (ImGui::TreeNodeEx(
group->name, ImGuiTreeNodeFlags_DefaultOpen)) {
549 for (
int node_idx = 0; node_idx <
nodes.Size; node_idx++) {
551 if (node->
GroupID != groupId) {
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;
560 node_selected = node->
ID;
562 if (ImGui::IsItemHovered()) {
563 node_hovered_in_list = node->
ID;
564 open_context_menu |= ImGui::IsMouseClicked(1);
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));
583 ImGui::PushStyleColor(ImGuiCol_WindowBg, (ImU32)ImColor(60, 60, 70, 200));
585 ImVec2 graphSize = ImGui::GetWindowSize();
586 if (
state.leftPaneVisible) {
587 graphSize.x -=
state.leftPaneSize;
589 if (
state.rightPaneVisible) {
590 graphSize.x -=
state.rightPaneSize;
592 graphSize.y -= toolbarSize.y + 20;
593 ImGui::BeginChild(
"scrolling_region", graphSize,
true, ImGuiWindowFlags_NoScrollbar | ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoScrollWithMouse);
594 ImGui::PushItemWidth(graphSize.x);
596 ImVec2
offset = ImGui::GetCursorScreenPos() - scrolling;
597 ImDrawList* draw_list = ImGui::GetWindowDrawList();
600 draw_list->ChannelsSplit((
nodes.Size + 2) * 2);
605 ImVec2 win_pos = ImGui::GetCursorScreenPos();
606 ImVec2 canvas_sz = ImGui::GetWindowSize();
608 for (
int link_idx = 0; link_idx < links.Size; link_idx++) {
610 NodeLink const& link = links[link_idx];
614 if ((
p1.x > win_pos.x + canvas_sz.x + 50) && (
p2.x > win_pos.x + canvas_sz.x + 50)) {
618 if ((
p1.y > win_pos.y + canvas_sz.y + 50) && (
p2.y > win_pos.y + canvas_sz.y + 50)) {
622 if ((
p1.y < win_pos.y) && (
p2.y < win_pos.y)) {
626 if ((
p1.x < win_pos.x) && (
p2.x < win_pos.x)) {
629 draw_list->ChannelsSetCurrent(0);
630 auto color = ARROW_BACKGROUND_COLOR;
631 auto thickness = ARROW_BACKGROUND_THICKNESS;
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)) {
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)) {
643 if (p1Inside && p2Inside) {
645 draw_list->ChannelsSetCurrent(2);
647 thickness = ARROW_THICKNESS;
648 }
else if (p1Inside || p2Inside) {
649 draw_list->ChannelsSetCurrent(1);
651 color = ARROW_HALFGROUND_COLOR;
652 thickness = ARROW_HALFGROUND_THICKNESS;
657 auto foregroundLayer = (
nodes.Size + 1) * 2 + 1;
658 draw_list->ChannelsSetCurrent(foregroundLayer);
659 color = ARROW_SELECTED_COLOR;
660 thickness = thickness + 2;
663 draw_list->AddBezierCurve(
p1,
p1 + ImVec2(+50, 0),
p2 + ImVec2(-50, 0),
p2,
color, thickness);
666 auto fgDrawList = ImGui::GetForegroundDrawList();
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;
672 if (node_selected == node_idx) {
673 backgroundLayer = (
nodes.Size + 1) * 2;
674 foregroundLayer = (
nodes.Size + 1) * 2 + 1;
680 ImVec2 node_rect_min =
offset +
pos->pos;
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)) {
688 ImGui::PushID(node->
ID);
691 draw_list->ChannelsSetCurrent(foregroundLayer);
692 bool old_any_active = ImGui::IsAnyItemActive();
695 ImGui::TextUnformatted(node->
Name);
697 case LogLevel::Critical:
699 ImGui::TextColored(ERROR_MESSAGE_COLOR,
"%s", ICON_FA_EXCLAMATION_CIRCLE);
701 case LogLevel::Error:
703 ImGui::TextColored(ERROR_MESSAGE_COLOR,
"%s", ICON_FA_EXCLAMATION_CIRCLE);
705 case LogLevel::Alarm:
707 ImGui::TextColored(WARNING_MESSAGE_COLOR,
"%s", ICON_FA_EXCLAMATION_TRIANGLE);
709 case LogLevel::Warning:
711 ImGui::TextColored(WARNING_MESSAGE_COLOR,
"%s", ICON_FA_EXCLAMATION_TRIANGLE);
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);
725 ImVec2 node_rect_max = node_rect_min + node->
Size;
726 ImVec2 node_rect_title = node_rect_min + ImVec2(node->
Size.x, 24);
738 draw_list->ChannelsSetCurrent(backgroundLayer);
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;
748 bool node_moving_active = ImGui::IsItemActive();
749 if (node_widgets_active || node_moving_active) {
750 node_selected = node->
ID;
752 if (node_moving_active && ImGui::IsMouseDragging(0)) {
753 pos->pos =
pos->pos + ImGui::GetIO().MouseDelta;
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);
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);
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);
772 for (
int slot_idx = 0; slot_idx < node->
InputsCount; slot_idx++) {
773 draw_list->ChannelsSetCurrent(backgroundLayer);
778 auto pp3 = p3 +
offset + slotPos;
779 auto color = ARROW_COLOR;
780 if (node_idx == node_selected) {
781 color = ARROW_SELECTED_COLOR;
783 draw_list->AddTriangleFilled(pp1, pp2, pp3,
color);
787 draw_list->ChannelsSetCurrent(foregroundLayer);
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),
811 label.textSize = ImGui::CalcTextSize(
label.buffer).x;
815 draw_list->AddText(
nullptr, 12,
817 NODE_LABEL_TEXT_COLOR,
821 for (
int slot_idx = 0; slot_idx < node->
OutputsCount; slot_idx++) {
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),
846 label.textSize = ImGui::CalcTextSize(
label.buffer).x;
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,
860 draw_list->ChannelsMerge();
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;
868 if (open_context_menu) {
869 ImGui::OpenPopup(
"context_menu");
870 if (node_hovered_in_list != -1) {
871 node_selected = node_hovered_in_list;
873 if (node_hovered_in_scene != -1) {
874 node_selected = node_hovered_in_scene;
882 ImGui::PopItemWidth();
884 ImGui::PopStyleColor();
885 ImGui::PopStyleVar(2);
888 if (
state.rightPaneVisible) {
891 ImGui::BeginChild(
"inspector");
892 ImGui::TextUnformatted(
"Device Inspector");
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];
902 auto& metadatum = metadata[
group.metadataId];
904 if (
state.rightPaneVisible) {
908 ImGui::TextWrapped(
"Select a node in the topology to display information about it");