mirror of
https://github.com/MadeOfJelly/MushMachine.git
synced 2025-04-19 01:32:58 +02:00
update strategy debug taks graph
This commit is contained in:
parent
d22ad0d913
commit
4f859bad2e
@ -7,9 +7,16 @@
|
|||||||
#include <vector>
|
#include <vector>
|
||||||
#include <set>
|
#include <set>
|
||||||
|
|
||||||
|
// fwd
|
||||||
|
namespace MM::Services {
|
||||||
|
class ImGuiEngineTools;
|
||||||
|
}
|
||||||
|
|
||||||
namespace MM::UpdateStrategies {
|
namespace MM::UpdateStrategies {
|
||||||
|
|
||||||
class SingleThreadedDefault : public MM::UpdateStrategies::UpdateStrategy {
|
class SingleThreadedDefault : public MM::UpdateStrategies::UpdateStrategy {
|
||||||
|
friend MM::Services::ImGuiEngineTools;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
struct Task {
|
struct Task {
|
||||||
std::string name;
|
std::string name;
|
||||||
|
@ -1,124 +1,345 @@
|
|||||||
#include "./engine_tools.hpp"
|
#include "./engine_tools.hpp"
|
||||||
|
#include "mm/update_strategies/update_strategy.hpp"
|
||||||
|
|
||||||
#include <entt/config/version.h>
|
#include <entt/config/version.h>
|
||||||
|
|
||||||
#include <mm/engine.hpp>
|
#include <mm/engine.hpp>
|
||||||
#include <entt/core/hashed_string.hpp>
|
#include <entt/core/hashed_string.hpp>
|
||||||
|
|
||||||
|
#include <mm/update_strategies/default_strategy.hpp>
|
||||||
|
|
||||||
|
#include <list>
|
||||||
|
|
||||||
#include <imgui/imgui.h>
|
#include <imgui/imgui.h>
|
||||||
|
|
||||||
namespace MM::Services {
|
namespace MM::Services {
|
||||||
|
|
||||||
bool ImGuiEngineTools::enable(Engine& engine) {
|
using UpdateStrategies::update_key_t;
|
||||||
auto& menu_bar = engine.getService<MM::Services::ImGuiMenuBar>();
|
|
||||||
|
|
||||||
menu_bar.menu_tree["Engine"]["Stop Engine"] = [](Engine& e) {
|
template<typename NColFn, typename LColFn, typename TTFn>
|
||||||
ImGui::Separator();
|
static void renderUpdateStratGraph(const std::unordered_map<update_key_t, std::set<update_key_t>>& g, std::set<update_key_t>& a, NColFn&& node_color_fn, LColFn&& line_color_fn, TTFn&& tooltip_fn) {
|
||||||
if (ImGui::MenuItem("Stop Engine")) {
|
//auto& g = us_default->getGraph(UpdateStrategies::update_phase_t::MAIN);
|
||||||
e.stop();
|
//auto& a = us_default->getActiveSet(UpdateStrategies::update_phase_t::MAIN);
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
menu_bar.menu_tree["Engine"]["About"] = [this](Engine&) {
|
using UpdateStrategies::update_key_t;
|
||||||
ImGui::MenuItem("About", NULL, &_show_about);
|
|
||||||
};
|
|
||||||
|
|
||||||
menu_bar.menu_tree["Engine"]["Services"] = [this](Engine&) {
|
std::set<update_key_t> work_queue{a.begin(), a.end()};
|
||||||
ImGui::MenuItem("Services", NULL, &_show_services);
|
|
||||||
};
|
|
||||||
|
|
||||||
return true;
|
std::vector<std::vector<update_key_t>> node_table{};
|
||||||
}
|
node_table.push_back({});
|
||||||
|
|
||||||
void ImGuiEngineTools::disable(Engine& engine) {
|
std::unordered_map<update_key_t, std::pair<size_t, size_t>> node_table_pos;
|
||||||
auto& menu_bar = engine.getService<MM::Services::ImGuiMenuBar>();
|
|
||||||
|
|
||||||
menu_bar.menu_tree["Engine"].erase("Stop Engine");
|
// THX to the schwammal
|
||||||
menu_bar.menu_tree["Engine"].erase("About");
|
|
||||||
menu_bar.menu_tree["Engine"].erase("Services");
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<UpdateStrategies::UpdateCreationInfo> ImGuiEngineTools::registerUpdates(void) {
|
{ // phase one, extract dep less nodes
|
||||||
using namespace entt::literals;
|
for (auto it = work_queue.begin(); it != work_queue.end();) {
|
||||||
return {
|
bool no_deps = g.at(*it).empty();
|
||||||
{
|
|
||||||
"ImGuiEngineTools::render"_hs,
|
if (!no_deps) {
|
||||||
"ImGuiEngineTools::render",
|
for (auto dep : g.at(*it)) {
|
||||||
[this](Engine& e){ renderImGui(e); },
|
if (a.count(dep)) {
|
||||||
UpdateStrategies::update_phase_t::MAIN,
|
no_deps = false;
|
||||||
true,
|
break;
|
||||||
{
|
}
|
||||||
"ImGuiMenuBar::render"_hs
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
void ImGuiEngineTools::renderImGui(Engine& engine) {
|
if (no_deps) {
|
||||||
if (_show_about) {
|
node_table_pos[*it] = {0, node_table[0].size()};
|
||||||
renderAbout();
|
node_table[0].emplace_back(*it);
|
||||||
}
|
it = work_queue.erase(it);
|
||||||
|
} else {
|
||||||
if (_show_services) {
|
it++;
|
||||||
renderServices(engine);
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void ImGuiEngineTools::renderAbout(void) {
|
{ // phase two, do rest
|
||||||
if (ImGui::Begin("About##EngineTools", &_show_about)) {
|
size_t curr_col = 0;
|
||||||
ImGui::Text("TODO");
|
while (!work_queue.empty()) {
|
||||||
ImGui::Text("MushMachine");
|
curr_col++;
|
||||||
ImGui::Text("EnTT v%d.%d.%d", ENTT_VERSION_MAJOR, ENTT_VERSION_MINOR, ENTT_VERSION_PATCH);
|
node_table.push_back({});
|
||||||
}
|
|
||||||
ImGui::End();
|
|
||||||
}
|
|
||||||
|
|
||||||
void ImGuiEngineTools::renderServices(Engine& engine) {
|
std::vector<update_key_t> nodes_this_round{};
|
||||||
if (ImGui::Begin("Services##EngineTools", &_show_services)) {
|
for (auto it = work_queue.begin(); it != work_queue.end();) {
|
||||||
ImGui::Text("TODO: make sortable");
|
bool deps_resolved = true;
|
||||||
ImGui::Checkbox("edit mode", &_services_edit_mode);
|
const auto& it_deps = g.at(*it);
|
||||||
|
|
||||||
if (ImGui::BeginTable(
|
for (const update_key_t dep : it_deps) {
|
||||||
"services_table",
|
if (a.count(dep) && work_queue.count(dep)) {
|
||||||
3,
|
deps_resolved = false;
|
||||||
ImGuiTableFlags_RowBg //|
|
break;
|
||||||
)) {
|
|
||||||
ImGui::TableSetupColumn("id", ImGuiTableColumnFlags_WidthFixed);
|
|
||||||
ImGui::TableSetupColumn("name");
|
|
||||||
ImGui::TableSetupColumn("status");
|
|
||||||
ImGui::TableHeadersRow();
|
|
||||||
|
|
||||||
for (auto& it : engine._services) {
|
|
||||||
ImGui::TableNextRow();
|
|
||||||
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::Text("%d", it.first);
|
|
||||||
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
ImGui::Text("%s", it.second->second->name());
|
|
||||||
|
|
||||||
ImGui::TableNextColumn();
|
|
||||||
if (_services_edit_mode) {
|
|
||||||
ImGui::PushID(it.first);
|
|
||||||
if (ImGui::SmallButton(it.second->first ? "enabled" : "disabled")) {
|
|
||||||
if (it.second->first) {
|
|
||||||
engine.disableService(it.first);
|
|
||||||
} else {
|
|
||||||
engine.enableService(it.first);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
//ImGui::SetTooltip("click to toggle!"); the heck?
|
|
||||||
ImGui::PopID();
|
|
||||||
} else {
|
|
||||||
ImGui::Text("%s", it.second->first ? "enabled" : "disabled");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ImGui::EndTable();
|
if (deps_resolved) {
|
||||||
|
//node_table_pos[*it] = {curr_col, node_table[curr_col].size()};
|
||||||
|
//node_table[curr_col].emplace_back(*it);
|
||||||
|
//it = work_queue.erase(it);
|
||||||
|
nodes_this_round.push_back(*it);
|
||||||
|
} else {
|
||||||
|
//it++;
|
||||||
|
}
|
||||||
|
it++;
|
||||||
|
}
|
||||||
|
|
||||||
|
// apply nodes
|
||||||
|
for (auto it : nodes_this_round) {
|
||||||
|
node_table_pos[it] = {curr_col, node_table[curr_col].size()};
|
||||||
|
node_table[curr_col].emplace_back(it);
|
||||||
|
work_queue.erase(it);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ImGui::End();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
{ // display
|
||||||
|
//ImGui::BeginChild("canvas", {0, 0}, true);
|
||||||
|
|
||||||
|
const ImVec2 cp = ImGui::GetCursorScreenPos();
|
||||||
|
const auto max_cont = ImGui::GetWindowContentRegionMax();
|
||||||
|
ImGui::Dummy(max_cont);
|
||||||
|
|
||||||
|
auto* dl = ImGui::GetWindowDrawList();
|
||||||
|
const float circle_radius = 20.f;
|
||||||
|
const float circle_padding = 40.f;
|
||||||
|
|
||||||
|
// lines
|
||||||
|
for (const update_key_t node : a) {
|
||||||
|
for (const update_key_t dep : g.at(node)) {
|
||||||
|
if (!a.count(dep)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
ImVec2 p_start{
|
||||||
|
cp.x + node_table_pos[node].first*(2*circle_radius+circle_padding) + circle_radius,
|
||||||
|
cp.y + node_table_pos[node].second*(2*circle_radius+circle_padding) + circle_radius,
|
||||||
|
};
|
||||||
|
|
||||||
|
ImVec2 p_end{
|
||||||
|
cp.x + node_table_pos[dep].first*(2*circle_radius+circle_padding) + circle_radius,
|
||||||
|
cp.y + node_table_pos[dep].second*(2*circle_radius+circle_padding) + circle_radius,
|
||||||
|
};
|
||||||
|
|
||||||
|
dl->AddLine(p_start, p_end, ImGui::ColorConvertFloat4ToU32(line_color_fn(node, dep)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// nodes
|
||||||
|
for (size_t curr_col = 0; curr_col < node_table.size(); curr_col++) {
|
||||||
|
const auto& curr_table_col = node_table[curr_col];
|
||||||
|
for (size_t curr_row = 0; curr_row < curr_table_col.size(); curr_row++) {
|
||||||
|
ImVec2 p{
|
||||||
|
cp.x + curr_col*(2*circle_radius+circle_padding) + circle_radius,
|
||||||
|
cp.y + curr_row*(2*circle_radius+circle_padding) + circle_radius,
|
||||||
|
};
|
||||||
|
|
||||||
|
//dl->AddCircleFilled(p, circle_radius, IM_COL32(100, 100, 100, 100));
|
||||||
|
auto col_vec = node_color_fn(curr_table_col.at(curr_row));
|
||||||
|
ImVec4 col_vec_bg = col_vec;
|
||||||
|
col_vec_bg.w *= 0.2f;
|
||||||
|
|
||||||
|
dl->AddCircleFilled(p, circle_radius, ImGui::ColorConvertFloat4ToU32(col_vec_bg));
|
||||||
|
dl->AddCircle(p, circle_radius, ImGui::ColorConvertFloat4ToU32(col_vec));
|
||||||
|
|
||||||
|
if (ImGui::IsMouseHoveringRect(
|
||||||
|
{p.x - circle_radius, p.y - circle_radius},
|
||||||
|
{p.x + circle_radius, p.y + circle_radius}
|
||||||
|
)) {
|
||||||
|
//ImGui::SetTooltip("'%s'\n[%u]", , curr_table_col.at(curr_row));
|
||||||
|
tooltip_fn(curr_table_col.at(curr_row));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//ImGui::EndChild();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ImGuiEngineTools::enable(Engine& engine) {
|
||||||
|
auto& menu_bar = engine.getService<MM::Services::ImGuiMenuBar>();
|
||||||
|
|
||||||
|
// use underscore hack to make it last
|
||||||
|
menu_bar.menu_tree["Engine"]["_Stop Engine"] = [](Engine& e) {
|
||||||
|
ImGui::Separator();
|
||||||
|
if (ImGui::MenuItem("Stop Engine")) {
|
||||||
|
e.stop();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
menu_bar.menu_tree["Engine"]["About"] = [this](Engine&) {
|
||||||
|
ImGui::MenuItem("About", NULL, &_show_about);
|
||||||
|
};
|
||||||
|
|
||||||
|
menu_bar.menu_tree["Engine"]["Services"] = [this](Engine&) {
|
||||||
|
ImGui::MenuItem("Services", NULL, &_show_services);
|
||||||
|
};
|
||||||
|
|
||||||
|
menu_bar.menu_tree["Engine"]["UpdateStrategy"] = [this](Engine&) {
|
||||||
|
ImGui::MenuItem("UpdateStrategy", NULL, &_show_update_stategy);
|
||||||
|
};
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ImGuiEngineTools::disable(Engine& engine) {
|
||||||
|
auto& menu_bar = engine.getService<MM::Services::ImGuiMenuBar>();
|
||||||
|
|
||||||
|
menu_bar.menu_tree["Engine"].erase("_Stop Engine");
|
||||||
|
menu_bar.menu_tree["Engine"].erase("About");
|
||||||
|
menu_bar.menu_tree["Engine"].erase("Services");
|
||||||
|
menu_bar.menu_tree["Engine"].erase("UpdateStrategy");
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<UpdateStrategies::UpdateCreationInfo> ImGuiEngineTools::registerUpdates(void) {
|
||||||
|
using namespace entt::literals;
|
||||||
|
return {
|
||||||
|
{
|
||||||
|
"ImGuiEngineTools::render"_hs,
|
||||||
|
"ImGuiEngineTools::render",
|
||||||
|
[this](Engine& e){ renderImGui(e); },
|
||||||
|
UpdateStrategies::update_phase_t::MAIN,
|
||||||
|
true,
|
||||||
|
{
|
||||||
|
"ImGuiMenuBar::render"_hs
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
void ImGuiEngineTools::renderImGui(Engine& engine) {
|
||||||
|
if (_show_about) {
|
||||||
|
renderAbout(engine);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (_show_services) {
|
||||||
|
renderServices(engine);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (_show_update_stategy) {
|
||||||
|
renderUpdateStrategy(engine);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ImGuiEngineTools::renderAbout(Engine& engine) {
|
||||||
|
if (ImGui::Begin("About##EngineTools", &_show_about)) {
|
||||||
|
ImGui::Text("TODO");
|
||||||
|
ImGui::Text("MushMachine");
|
||||||
|
ImGui::Text("UpdateStrategy: %s", engine.getUpdateStrategy().name());
|
||||||
|
ImGui::Text("EnTT v%d.%d.%d", ENTT_VERSION_MAJOR, ENTT_VERSION_MINOR, ENTT_VERSION_PATCH);
|
||||||
|
}
|
||||||
|
ImGui::End();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ImGuiEngineTools::renderServices(Engine& engine) {
|
||||||
|
if (ImGui::Begin("Services##EngineTools", &_show_services)) {
|
||||||
|
ImGui::Text("TODO: make sortable");
|
||||||
|
ImGui::Checkbox("edit mode", &_services_edit_mode);
|
||||||
|
|
||||||
|
if (ImGui::BeginTable(
|
||||||
|
"services_table",
|
||||||
|
3,
|
||||||
|
ImGuiTableFlags_RowBg //|
|
||||||
|
)) {
|
||||||
|
ImGui::TableSetupColumn("id", ImGuiTableColumnFlags_WidthFixed);
|
||||||
|
ImGui::TableSetupColumn("name");
|
||||||
|
ImGui::TableSetupColumn("status");
|
||||||
|
ImGui::TableHeadersRow();
|
||||||
|
|
||||||
|
for (auto& it : engine._services) {
|
||||||
|
ImGui::TableNextRow();
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("%d", it.first);
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("%s", it.second->second->name());
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
if (_services_edit_mode) {
|
||||||
|
ImGui::PushID(it.first);
|
||||||
|
if (ImGui::SmallButton(it.second->first ? "enabled" : "disabled")) {
|
||||||
|
if (it.second->first) {
|
||||||
|
engine.disableService(it.first);
|
||||||
|
} else {
|
||||||
|
engine.enableService(it.first);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
//ImGui::SetTooltip("click to toggle!"); the heck?
|
||||||
|
ImGui::PopID();
|
||||||
|
} else {
|
||||||
|
ImGui::Text("%s", it.second->first ? "enabled" : "disabled");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::EndTable();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ImGui::End();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ImGuiEngineTools::renderUpdateStrategy(Engine& engine) {
|
||||||
|
if (ImGui::Begin("UpdateStrategy##EngineTools", &_show_update_stategy, ImGuiWindowFlags_NoScrollbar | ImGuiWindowFlags_NoScrollWithMouse)) {
|
||||||
|
auto& us = engine.getUpdateStrategy();
|
||||||
|
auto us_name = us.name();
|
||||||
|
|
||||||
|
ImGui::Text("UpdateStrategy: '%s'", us_name);
|
||||||
|
ImGui::Separator();
|
||||||
|
|
||||||
|
auto* us_default = dynamic_cast<MM::UpdateStrategies::SingleThreadedDefault*>(&us);
|
||||||
|
if (us_default) {
|
||||||
|
auto& g = us_default->getGraph(UpdateStrategies::update_phase_t::MAIN);
|
||||||
|
auto& a = us_default->getActiveSet(UpdateStrategies::update_phase_t::MAIN);
|
||||||
|
std::set<update_key_t> nodes;
|
||||||
|
for (const auto& it : g) {
|
||||||
|
nodes.emplace(it.first);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ImGui::BeginTabBar("tabs")) {
|
||||||
|
if (ImGui::BeginTabItem("table")) {
|
||||||
|
if (ImGui::BeginTable("table", 2)) {
|
||||||
|
for (const auto it : nodes) {
|
||||||
|
ImGui::TableNextRow();
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("%u", it);
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("%s", us_default->_tasks[it].name.c_str());
|
||||||
|
}
|
||||||
|
ImGui::EndTable();
|
||||||
|
}
|
||||||
|
ImGui::EndTabItem();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ImGui::BeginTabItem("graph")) {
|
||||||
|
renderUpdateStratGraph(g, nodes,
|
||||||
|
[&a](const update_key_t key) -> ImVec4 {
|
||||||
|
if (a.count(key)) {
|
||||||
|
return {0.9f, 1.f, 0.9f, 1.f};
|
||||||
|
} else {
|
||||||
|
return {0.4f, 0.2f, 0.2f, 1.f};
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[&a](const update_key_t from, const update_key_t to) -> ImVec4 {
|
||||||
|
if (a.count(from) && a.count(to)) {
|
||||||
|
return {0.9f, 1.f, 0.9f, 1.f};
|
||||||
|
} else {
|
||||||
|
return {0.4f, 0.2f, 0.2f, 1.f};
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[us_default](update_key_t key) {
|
||||||
|
ImGui::SetTooltip("'%s'\n[%u]", us_default->_tasks[key].name.c_str(), key);
|
||||||
|
});
|
||||||
|
ImGui::EndTabItem();
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::EndTabBar();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ImGui::End();
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace MM::Services
|
} // namespace MM::Services
|
||||||
|
|
||||||
|
@ -16,14 +16,16 @@ namespace MM::Services {
|
|||||||
private:
|
private:
|
||||||
bool _show_about = false;
|
bool _show_about = false;
|
||||||
bool _show_services = false;
|
bool _show_services = false;
|
||||||
|
bool _show_update_stategy = false;
|
||||||
|
|
||||||
bool _services_edit_mode = false;
|
bool _services_edit_mode = false;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void renderImGui(Engine& engine);
|
void renderImGui(Engine& engine);
|
||||||
|
|
||||||
void renderAbout(void);
|
void renderAbout(Engine& engine);
|
||||||
void renderServices(Engine& engine);
|
void renderServices(Engine& engine);
|
||||||
|
void renderUpdateStrategy(Engine& engine);
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace MM::Services
|
} // namespace MM::Services
|
||||||
|
@ -44,7 +44,8 @@ std::vector<UpdateStrategies::UpdateCreationInfo> SimpleSceneService::registerUp
|
|||||||
[this](Engine& e) { this->changeSceneFixedUpdate(e); },
|
[this](Engine& e) { this->changeSceneFixedUpdate(e); },
|
||||||
UpdateStrategies::update_phase_t::MAIN,
|
UpdateStrategies::update_phase_t::MAIN,
|
||||||
true,
|
true,
|
||||||
{"SimpleSceneService::scene_update"_hs} // first update, than change????
|
//{"SimpleSceneService::scene_update"_hs} // first update, than change????
|
||||||
|
{"SimpleSceneService::scene_tick"_hs}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user