/** * @file main.cpp * The main file, where it all happens. */ // standard library includes #include using namespace std::literals::chrono_literals; // local library includes #include // our own includes #include #include #include #include #include #include #include #include #include #include #include #include /** * The currently used folder to look for XML files in. */ std::string xmlFolder = "xml/"; /** * The current center of the screen, updated in main render. */ vec2 offset; /** * The main program. * Init, load, run. Die. */ int main(int argc, char *argv[]) { static bool worldReset = false, worldDontReallyRun = false; std::string worldActuallyUseThisXMLFile; // handle command line args if (argc > 1) { for (int i = 1; i < argc; i++) { std::string s = argv[i]; if (s == "--reset" || s == "-r") worldReset = true; else if (s == "--dontrun" || s == "-d") worldDontReallyRun = true; else if (s == "--xml" || s == "-x") worldActuallyUseThisXMLFile = argv[i + 1]; } } // // init the main game engine // game::engine.init(); // initialize the renderer Render::init(); // start the random number generator randInit(millis()); // load some saved data game::briceLoad(); game::briceUpdate(); // read in all XML file names in the folder std::list xmlFiles; if (getdir(std::string("./" + xmlFolder), xmlFiles)) UserError("Error reading XML files!!!"); // kill the world if needed if (worldReset) { // TODO TODO TODO we do xml/*.dat now... kill that game::briceClear(); } // either load the given XML, or find one auto worldSys = game::engine.getSystem(); if (!worldActuallyUseThisXMLFile.empty()) { worldSys->load(worldActuallyUseThisXMLFile); } else { // load the first valid XML file for the world for (const auto &xf : xmlFiles) { if (xf[0] != '.') { // read it in std::cout << "File to load: " << xf << '\n'; worldSys->load(xf); break; } } } ///////////////////////////// // // // actually start the game // // // ///////////////////////////// game::engine.getSystem()->add("Debug", 9); std::list eventQueue; if (!worldDontReallyRun) { // the main loop, in all of its gloriousness... GameThread gtMain ([&] { game::time::mainLoopHandler(); if (game::time::tickHasPassed()) { // calculate the world shading value extern int worldShade; // TODO kill worldShade = 50 * sin((game::time::getTickCount() + (DAY_CYCLE / 2)) / (DAY_CYCLE / PI)); // update fades //ui::fadeUpdate(); // increment game ticker game::time::tick(); } while (!eventQueue.empty()) { game::events.emit(eventQueue.back()); eventQueue.pop_back(); } game::engine.update(game::time::getDeltaTime()); std::this_thread::sleep_for(1ms); }); static int fps = 0, fpsInternal = 0; // the debug loop, gets debug screen values GameThread gtDebug ([&] { fps = fpsInternal, fpsInternal = 0; std::this_thread::sleep_for(1s); }); GameThread gtFade ([&] { ui::fadeUpdate(); std::this_thread::sleep_for(20ms); }); // the render loop, renders const bool &run = game::engine.shouldRun; while (run) { fpsInternal++; Render::render(fps); SDL_Event e; while (SDL_PollEvent(&e)) eventQueue.push_back(e); } // on game end, get back together gtMain.stop(); gtDebug.stop(); gtFade.stop(); //game::engine.getSystem()->thAmbient.join(); // segfault or something } // save game::briceSave(); worldSys->save(); // exit Mix_HaltMusic(); Mix_CloseAudio(); ui::destroyFonts(); unloadTextures(); game::engine.getSystem()->die(); return 0; // Calls everything passed to atexit }