#include "./main_screen.hpp" #include #include #include #include MainScreen::MainScreen(SDL_Renderer* renderer_, std::string save_path, std::string save_password, std::vector plugins) : renderer(renderer_), rmm(cr), mts(rmm), tc(save_path, save_password), tpi(tc.getTox()), ad(tc), tcm(cr, tc, tc), tmm(rmm, cr, tcm, tc, tc), ttm(rmm, cr, tcm, tc, tc), tffom(cr, rmm, tcm, tc, tc), mmil(rmm), tam(rmm, cr, conf), sdlrtu(renderer_), cg(conf, rmm, cr, sdlrtu), sw(conf), tuiu(tc, conf), tdch(tpi) { tel.subscribeAll(tc); conf.set("tox", "save_file_path", save_path); { // name stuff auto name = tc.toxSelfGetName(); if (name.empty()) { name = "tomato"; } conf.set("tox", "name", name); tc.setSelfName(name); // TODO: this is ugly } // TODO: remove std::cout << "own address: " << tc.toxSelfGetAddressStr() << "\n"; { // setup plugin instances g_provideInstance("ConfigModelI", "host", &conf); g_provideInstance("Contact3Registry", "1", "host", &cr); g_provideInstance("RegistryMessageModel", "host", &rmm); g_provideInstance("ToxI", "host", &tc); g_provideInstance("ToxPrivateI", "host", &tpi); g_provideInstance("ToxEventProviderI", "host", &tc); g_provideInstance("ToxContactModel2", "host", &tcm); // TODO: pm? // graphics g_provideInstance("ImGuiContext", ImGui::GetVersion(), "host", ImGui::GetCurrentContext()); g_provideInstance("TextureUploaderI", "host", &sdlrtu); } for (const auto& ppath : plugins) { if (!pm.add(ppath)) { std::cerr << "MS error: loading plugin '" << ppath << "' failed!\n"; // thow? assert(false && "failed to load plugin"); } } conf.dump(); } MainScreen::~MainScreen(void) { } bool MainScreen::handleEvent(SDL_Event& e) { if (e.type == SDL_EVENT_DROP_FILE) { std::cout << "DROP FILE: " << e.drop.data << "\n"; cg.sendFilePath(e.drop.data); _render_interval = 1.f/60.f; // TODO: magic _time_since_event = 0.f; return true; // TODO: forward return succ from sendFilePath() } if ( e.type == SDL_EVENT_WINDOW_MINIMIZED || e.type == SDL_EVENT_WINDOW_HIDDEN || e.type == SDL_EVENT_WINDOW_OCCLUDED // does this trigger on partial occlusion? ) { auto* window = SDL_GetWindowFromID(e.window.windowID); auto* event_renderer = SDL_GetRenderer(window); if (event_renderer != nullptr && event_renderer == renderer) { // our window is now obstructed if (_window_hidden_ts < e.window.timestamp) { _window_hidden_ts = e.window.timestamp; _window_hidden = true; //std::cout << "TOMAT: window hidden " << e.window.timestamp << "\n"; } } return true; // forward? } if ( e.type == SDL_EVENT_WINDOW_SHOWN || e.type == SDL_EVENT_WINDOW_RESTORED || e.type == SDL_EVENT_WINDOW_EXPOSED ) { auto* window = SDL_GetWindowFromID(e.window.windowID); auto* event_renderer = SDL_GetRenderer(window); if (event_renderer != nullptr && event_renderer == renderer) { if (_window_hidden_ts <= e.window.timestamp) { _window_hidden_ts = e.window.timestamp; if (_window_hidden) { // if window was previously hidden, we shorten the wait for the next frame _render_interval = 1.f/60.f; } _window_hidden = false; //std::cout << "TOMAT: window shown " << e.window.timestamp << "\n"; } } _render_interval = 1.f/60.f; // TODO: magic _time_since_event = 0.f; return true; // forward? } if ( _fps_perf_mode <= 1 && ( // those are all the events imgui polls e.type == SDL_EVENT_MOUSE_MOTION || e.type == SDL_EVENT_MOUSE_WHEEL || e.type == SDL_EVENT_MOUSE_BUTTON_DOWN || e.type == SDL_EVENT_MOUSE_BUTTON_UP || e.type == SDL_EVENT_TEXT_INPUT || e.type == SDL_EVENT_KEY_DOWN || e.type == SDL_EVENT_KEY_UP || e.type == SDL_EVENT_WINDOW_MOUSE_ENTER || e.type == SDL_EVENT_WINDOW_MOUSE_LEAVE || e.type == SDL_EVENT_WINDOW_FOCUS_GAINED || e.type == SDL_EVENT_WINDOW_FOCUS_LOST ) ) { _render_interval = 1.f/60.f; // TODO: magic _time_since_event = 0.f; } return false; } Screen* MainScreen::render(float time_delta, bool&) { // HACK: render the tomato main window first, with proper flags set. // flags need to be set the first time begin() is called. // and plugins are run before the main cg is run. { // TODO: maybe render cg earlier? or move the main window out of cg? constexpr auto bg_window_flags = ImGuiWindowFlags_NoDecoration | ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoResize | ImGuiWindowFlags_NoSavedSettings | ImGuiWindowFlags_MenuBar | ImGuiWindowFlags_NoBringToFrontOnFocus; ImGui::Begin("tomato", nullptr, bg_window_flags); ImGui::End(); } const float pm_interval = pm.render(time_delta); // render cg.render(time_delta); // render sw.render(); // render tuiu.render(); // render tdch.render(); // render { // main window menubar injection if (ImGui::Begin("tomato")) { if (ImGui::BeginMenuBar()) { // ImGui::Separator(); // why do we not need this???? if (ImGui::BeginMenu("Performance")) { { // fps const auto targets = "normal\0reduced\0powersave\0"; ImGui::SetNextItemWidth(ImGui::GetFontSize()*10); ImGui::Combo("fps mode", &_fps_perf_mode, targets, 4); } { // compute const auto targets = "normal\0powersave\0extreme powersave\0"; ImGui::SetNextItemWidth(ImGui::GetFontSize()*10); ImGui::Combo("compute mode", &_compute_perf_mode, targets, 4); ImGui::SetItemTooltip("Limiting compute can slow down things like filetransfers!"); } ImGui::EndMenu(); } if (ImGui::BeginMenu("Settings")) { if (ImGui::MenuItem("ImGui Style Editor")) { _show_tool_style_editor = true; } ImGui::EndMenu(); } ImGui::EndMenuBar(); } } ImGui::End(); } if (_show_tool_style_editor) { if (ImGui::Begin("Dear ImGui Style Editor", &_show_tool_style_editor)) { ImGui::ShowStyleEditor(); } ImGui::End(); } if constexpr (false) { ImGui::ShowDemoWindow(); } if ( _fps_perf_mode > 1 // TODO: magic ) { // powersave forces 250ms _render_interval = 1.f/4.f; } else if ( _time_since_event > 1.f && ( // 1sec cool down _fps_perf_mode == 1 || // TODO: magic _window_hidden ) ) { _render_interval = std::min(1.f/1.f, pm_interval); } else { _render_interval = std::min(1.f/60.f, pm_interval); } _time_since_event += time_delta; return nullptr; } Screen* MainScreen::tick(float time_delta, bool& quit) { quit = !tc.iterate(); // compute tcm.iterate(time_delta); // compute const float fo_interval = tffom.tick(time_delta); tam.iterate(); // compute const float pm_interval = pm.tick(time_delta); // compute tdch.tick(time_delta); // compute mts.iterate(); // compute _min_tick_interval = std::min( // HACK: pow by 1.6 to increase 50 -> ~500 (~522) // and it does not change 1 std::pow(tc.toxIterationInterval(), 1.6f)/1000.f, pm_interval ); _min_tick_interval = std::min( _min_tick_interval, fo_interval ); //std::cout << "MS: min tick interval: " << _min_tick_interval << "\n"; switch (_compute_perf_mode) { // normal 1ms lower bound case 0: _min_tick_interval = std::max(_min_tick_interval, 0.001f); break; // in powersave fix the lowerbound to 100ms case 1: _min_tick_interval = std::max(_min_tick_interval, 0.1f); break; // extreme 2s case 2: _min_tick_interval = std::max(_min_tick_interval, 2.f); break; default: std::cerr << "unknown compute perf mode\n"; std::exit(-1); } return nullptr; }