#include #include #include #include #include #include "./theme.hpp" #include "./chat_gui/theme.hpp" #include "./sys_check.hpp" #include "./start_screen.hpp" #include #include #include #include #include #include #ifdef TOMATO_BREAKPAD # include "./breakpad_client.hpp" #endif int main(int argc, char** argv) { runSysCheck(); #ifdef TOMATO_BREAKPAD // TODO: maybe run before sys check? BREAKPAD_MAIN_INIT; std::cout << "Breakpad handler installed.\n"; #endif // better args std::vector args; for (int i = 0; i < argc; i++) { args.push_back(argv[i]); } SDL_SetAppMetadata("tomato", "0.0.0-wip", nullptr); #ifdef __ANDROID__ // change current working dir to internal storage std::filesystem::current_path(SDL_GetAndroidInternalStoragePath()); #endif // setup hints #ifndef __ANDROID__ if (!SDL_SetHint(SDL_HINT_VIDEO_ALLOW_SCREENSAVER, "1")) { std::cerr << "Failed to set '" << SDL_HINT_VIDEO_ALLOW_SCREENSAVER << "' to 1\n"; } #endif auto last_time_render = std::chrono::steady_clock::now(); auto last_time_tick = std::chrono::steady_clock::now(); // actual setup if (!SDL_Init(SDL_INIT_VIDEO)) { std::cerr << "SDL_Init failed (" << SDL_GetError() << ")\n"; return 1; } // more RAII std::unique_ptr window { SDL_CreateWindow("tomato", 1280, 720, SDL_WINDOW_RESIZABLE | SDL_WINDOW_HIGH_PIXEL_DENSITY), &SDL_DestroyWindow }; if (!window) { std::cerr << "SDL_CreateWindow failed (" << SDL_GetError() << ")\n"; return 1; } std::unique_ptr renderer { SDL_CreateRenderer(window.get(), nullptr), &SDL_DestroyRenderer }; if (!renderer) { std::cerr << "SDL_CreateRenderer failed (" << SDL_GetError() << ")\n"; return 1; } SDL_SetRenderVSync(renderer.get(), SDL_RENDERER_VSYNC_ADAPTIVE); std::cout << "SDL Renderer: " << SDL_GetRendererName(renderer.get()) << "\n"; IMGUI_CHECKVERSION(); ImGui::CreateContext(); // TODO: test android behaviour float display_scale = SDL_GetWindowDisplayScale(window.get()); if (display_scale < 0.001f) { // error? display_scale = 1.f; } ImGui::GetStyle().ScaleAllSizes(display_scale); Theme theme; if (SDL_GetSystemTheme() == SDL_SYSTEM_THEME_LIGHT) { ImGui::StyleColorsLight(); theme = getDefaultThemeLight(); } else { //ImGui::StyleColorsDark(); setThemeGreen(); theme = getDefaultThemeDark(); } ImGui_ImplSDL3_InitForSDLRenderer(window.get(), renderer.get()); ImGui_ImplSDLRenderer3_Init(renderer.get()); std::unique_ptr screen = std::make_unique(args, renderer.get(), theme); bool is_background = false; bool quit = false; while (!quit) { auto new_time = std::chrono::steady_clock::now(); const float time_delta_tick = std::chrono::duration(new_time - last_time_tick).count(); const float time_delta_render = std::chrono::duration(new_time - last_time_render).count(); bool tick = time_delta_tick >= screen->nextTick(); bool render = time_delta_render >= screen->nextRender(); if (tick) { Screen* ret_screen = screen->tick(time_delta_tick, quit); if (ret_screen != nullptr) { screen.reset(ret_screen); } last_time_tick = new_time; } // do events outside of tick/render, so they can influence reported intervals SDL_Event event; while (SDL_PollEvent(&event)) { if (event.type == SDL_EVENT_QUIT) { quit = true; break; } else if (event.type == SDL_EVENT_WILL_ENTER_BACKGROUND) { is_background = true; } else if (event.type == SDL_EVENT_DID_ENTER_FOREGROUND) { is_background = false; } if (screen->handleEvent(event)) { continue; } ImGui_ImplSDL3_ProcessEvent(&event); } if (quit) { break; } if (is_background) { render = false; } // can do both in the same loop if (render) { ImGui_ImplSDLRenderer3_NewFrame(); ImGui_ImplSDL3_NewFrame(); ImGui::NewFrame(); { // render Screen* ret_screen = screen->render(time_delta_render, quit); if (ret_screen != nullptr) { screen.reset(ret_screen); } } ImGui::Render(); ImGui_ImplSDLRenderer3_RenderDrawData(ImGui::GetDrawData(), renderer.get()); SDL_RenderPresent(renderer.get()); // clearing after present is (should) more performant, but first frame is a mess SDL_SetRenderDrawColor(renderer.get(), 0x10, 0x10, 0x10, SDL_ALPHA_OPAQUE); SDL_RenderClear(renderer.get()); last_time_render = new_time; } //// TODO: seperate out render and tick //const float time_to_next_loop = std::min(screen->nextRender(), screen->nextTick()); //std::this_thread::sleep_for( // time left to get to 60fps //std::chrono::duration(time_to_next_loop) //- std::chrono::duration(std::chrono::steady_clock::now() - new_time) // time used for rendering //); #if 1 if (render || tick) { // why is windows like this //std::this_thread::sleep_for(std::chrono::milliseconds(1)); // yield for 1ms SDL_Delay(1); // yield for 1ms } else { #if 0 // pretty hacky and spins if close to next update // if next loop >= 1ms away, wait 1ms if (time_delta_tick+0.001f < screen->nextTick() && time_delta_render+0.001f < screen->nextRender()) { std::this_thread::sleep_for(std::chrono::milliseconds(1)); // yield for 1ms } #else // dynamic sleep, sleeps the reminder till next update //std::this_thread::sleep_for(std::chrono::duration( //std::min( //screen->nextTick() - time_delta_tick, //screen->nextRender() - time_delta_render //) //)); const float min_delay = std::min( std::min( screen->nextTick() - time_delta_tick, screen->nextRender() - time_delta_render ), 0.25f // dont sleep too long ) * 1000.f; if (min_delay > 0.f) { SDL_Delay(uint32_t(min_delay)); } // better in theory, but consumes more cpu on linux for some reason //SDL_WaitEventTimeout(nullptr, int32_t( //std::min( //screen->nextTick() - time_delta_tick, //screen->nextRender() - time_delta_render //) * 1000.f //)); #endif } #else // why is windows like this //std::this_thread::sleep_for(std::chrono::milliseconds(1)); // yield for 1ms SDL_Delay(1); // yield for 1ms #endif } // TODO: use scope for the unique ptrs screen.reset(); ImGui_ImplSDLRenderer3_Shutdown(); ImGui_ImplSDL3_Shutdown(); ImGui::DestroyContext(); renderer.reset(); window.reset(); SDL_Quit(); return 0; }