diff options
author | Andy <drumsetmonkey@gmail.com> | 2017-01-19 09:21:12 -0500 |
---|---|---|
committer | Andy <drumsetmonkey@gmail.com> | 2017-01-19 09:21:12 -0500 |
commit | 213d9ccfbb4752d4c62d6b7e6b3f9172cdf1bccc (patch) | |
tree | 7872c6f30c8adf048a7863a33d837299c7fb0771 /main.cpp | |
parent | 19a32074595a4a2797eaeb978f8bd302f736f6a6 (diff) | |
parent | 8452b199d28bea53bf2c5e3b3d604064000fc73d (diff) |
Limb animation actually works
Diffstat (limited to 'main.cpp')
-rw-r--r-- | main.cpp | 291 |
1 files changed, 72 insertions, 219 deletions
@@ -4,15 +4,11 @@ */ // standard library includes -#include <fstream> -#include <mutex> #include <chrono> using namespace std::literals::chrono_literals; // local library includes #include <entityx/entityx.h> -#include <tinyxml2.h> -using namespace tinyxml2; // our own includes #include <brice.hpp> @@ -20,17 +16,40 @@ using namespace tinyxml2; #include <common.hpp> #include <engine.hpp> #include <gametime.hpp> -#include <player.hpp> #include <window.hpp> #include <world.hpp> #include <render.hpp> #include <ui.hpp> -#include <particle.hpp> +#include <inventory.hpp> + +class GameThread : public entityx::Receiver<GameThread> { +private: + std::atomic_bool die; + std::thread thread; + +public: + template<typename F> + GameThread(F&& func) { + die.store(false); + thread = std::thread([this](F&& f) { + while (!die.load()) + f(); + }, std::forward<F>(func)); + } + + ~GameThread(void) { + thread.join(); + } + + void stop(void) { + die.store(true); + } +}; /** * The currently used folder to look for XML files in. */ -std::string xmlFolder; +std::string xmlFolder = "xml/"; /** * The current menu, if any are open (TODO why is this here) @@ -43,13 +62,6 @@ Menu *currentMenu; vec2 offset; /** - * The current FPS of the game. - */ -static unsigned int fps = 0; - -void render(void); - -/** * The main program. * Init, load, run. Die. */ @@ -58,10 +70,7 @@ int main(int argc, char *argv[]) static bool worldReset = false, worldDontReallyRun = false; std::string worldActuallyUseThisXMLFile; - // - // get command line arguments, if any - // - + // handle command line args if (argc > 1) { for (int i = 1; i < argc; i++) { std::string s = argv[i]; @@ -80,82 +89,30 @@ int main(int argc, char *argv[]) // game::engine.init(); - // used three times below - auto worldSys = game::engine.getSystem<WorldSystem>(); - - // - // initialize GLEW - // -#ifndef __WIN32__ - glewExperimental = GL_TRUE; -#endif - - auto glewError = glewInit(); - if (glewError != GLEW_OK) - UserError(std::string("GLEW was not able to initialize! Error: ") - + reinterpret_cast<const char *>(glewGetErrorString(glewError))); - - // - // start the random number generator (TODO our own?) - // + // initialize the renderer + Render::init(); + // start the random number generator randInit(millis()); - // - // some basic OpenGL setup stuff - // - - SDL_GL_SetAttribute(SDL_GL_DOUBLEBUFFER, 0); - // enable v-sync (TODO but 1000 fps?) - SDL_GL_SetSwapInterval(0); - // hide the cursor - SDL_ShowCursor(SDL_DISABLE); - // switch to pixel grid - glViewport(0, 0, game::SCREEN_WIDTH, game::SCREEN_HEIGHT); - glEnable(GL_BLEND); - glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); - glClearColor(1, 1, 1, 1); - - // - // initialize shaders - // - - std::cout << "Initializing shaders!\n"; - Render::initShaders(); - Colors::init(); - - // // load some saved data - // - game::briceLoad(); game::briceUpdate(); - // load sprites used in the inventory menu. See src/inventory.cpp - //initInventorySprites(); - - // get a world - // - - if (xmlFolder.empty()) - xmlFolder = "xml/"; - // read in all XML file names in the folder - std::vector<std::string> xmlFiles; - if (getdir(std::string("./" + xmlFolder).c_str(), xmlFiles)) + std::list<std::string> xmlFiles; + if (getdir(std::string("./" + xmlFolder), xmlFiles)) UserError("Error reading XML files!!!"); - // alphabetically sort files - strVectorSortAlpha(&xmlFiles); - // kill the world if needed if (worldReset) { - // TODO TODO TODO we do xml/*.dat now... + // 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<WorldSystem>(); if (!worldActuallyUseThisXMLFile.empty()) { worldSys->load(worldActuallyUseThisXMLFile); } else { @@ -170,80 +127,74 @@ int main(int argc, char *argv[]) } } - // - // initialize ui - // - - ui::menu::init(); - - ///////////////////////////// // // // actually start the game // // // ///////////////////////////// + + game::engine.getSystem<InventorySystem>()->add("Debug", 3); + + + std::list<SDL_Event> eventQueue; + if (!worldDontReallyRun) { // the main loop, in all of its gloriousness... - std::thread thMain ([&] { - const bool &run = game::engine.shouldRun; - while (run) { - 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)); + GameThread gtMain ([&] { + game::time::mainLoopHandler(); - // update fades - ui::fadeUpdate(); + 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)); - // increment game ticker - game::time::tick(); - } + // update fades + ui::fadeUpdate(); - game::engine.update(game::time::getDeltaTime()); + // increment game ticker + game::time::tick(); + } - std::this_thread::sleep_for(1ms); + while (!eventQueue.empty()) { + game::events.emit<MainSDLEvent>(eventQueue.back()); + eventQueue.pop_back(); } - }); - static float fpsInternal = 0; + 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 - std::thread thDebug ([&] { - const bool &run = game::engine.shouldRun; - while (run) { - fps = fpsInternal, fpsInternal = 0; - std::this_thread::sleep_for(1s); - } + GameThread gtDebug ([&] { + fps = fpsInternal, fpsInternal = 0; + std::this_thread::sleep_for(1s); }); - // thre render loop, renders + // the render loop, renders const bool &run = game::engine.shouldRun; while (run) { fpsInternal++; - render(); - game::engine.resetRender(0); // TODO stupid + Render::render(fps); + + SDL_Event e; + while (SDL_PollEvent(&e)) + eventQueue.push_back(e); } // on game end, get back together - thMain.join(); - thDebug.join(); + gtMain.stop(); + gtDebug.stop(); //game::engine.getSystem<WorldSystem>()->thAmbient.join(); // segfault or something } - // - // put away the brice for later, save world - // - + // save game::briceSave(); worldSys->save(); - // - // close things, free stuff, yada yada - // - + // exit Mix_HaltMusic(); Mix_CloseAudio(); @@ -252,103 +203,5 @@ int main(int argc, char *argv[]) game::engine.getSystem<WindowSystem>()->die(); - // - // goodbye - // - return 0; // Calls everything passed to atexit } - -void render() { - static const Texture mouseTex ("assets/goodmouse.png"); - static const glm::mat4 view = glm::lookAt( - glm::vec3(0.0f, 0.0f, 0.0f), // pos - glm::vec3(0.0f, 0.0f, -10.0f), // looking at - glm::vec3(0.0f, 1.0f, 0.0f) // up vector - ); - - static const auto& SCREEN_WIDTH2 = game::SCREEN_WIDTH / 2.0f; - static const auto& SCREEN_HEIGHT2 = game::SCREEN_HEIGHT / 2.0f; - - // - // set the ortho - // - - auto ps = game::engine.getSystem<PlayerSystem>(); - auto ploc = ps->getPosition(); - offset.x = ploc.x + ps->getWidth() / 2; - - const auto& worldWidth = game::engine.getSystem<WorldSystem>()->getWidth(); - if (worldWidth < (int)SCREEN_WIDTH2 * 2) - offset.x = 0; - else if (offset.x - SCREEN_WIDTH2 < worldWidth * -0.5f) - offset.x = ((worldWidth * -0.5f) + SCREEN_WIDTH2); - else if (offset.x + SCREEN_WIDTH2 > worldWidth * 0.5f) - offset.x = ((worldWidth * 0.5f) - SCREEN_WIDTH2); - - // ortho y snapping - offset.y = std::max(ploc.y /*+ player->height / 2*/, SCREEN_HEIGHT2); - - // "setup" - glm::mat4 projection = glm::ortho(floor(offset.x - SCREEN_WIDTH2), // left - floor(offset.x + SCREEN_WIDTH2), // right - floor(offset.y - SCREEN_HEIGHT2), // bottom - floor(offset.y + SCREEN_HEIGHT2), // top - static_cast<decltype(floor(10.0f))>(10.0), // near - static_cast<decltype(floor(10.0f))>(-10.0)); // far - - glm::mat4 ortho = projection * view; - - glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); - - // TODO add depth - glEnable(GL_DEPTH_TEST); - - - Render::worldShader.use(); - glUniformMatrix4fv(Render::worldShader.uniform[WU_ortho], 1, GL_FALSE, glm::value_ptr(ortho)); - glUniformMatrix4fv(Render::worldShader.uniform[WU_transform], 1, GL_FALSE, glm::value_ptr(glm::mat4(1.0f))); - Render::worldShader.unuse(); - - Render::textShader.use(); - glUniformMatrix4fv(Render::textShader.uniform[WU_transform], 1, GL_FALSE, glm::value_ptr(ortho)); - glUniform4f(Render::textShader.uniform[WU_tex_color], 1.0, 1.0, 1.0, 1.0); - Render::textShader.unuse(); - - // draw the world and player - game::engine.getSystem<WorldSystem>()->render(); - game::engine.getSystem<ParticleSystem>()->render(); - - // draw the player's inventory - //player->inv->draw(); - - game::engine.render(0); - - // draw the fade overlay - ui::drawFade(); - - // draw ui elements - ui::draw(); - - // draw the debug overlay if desired - if (ui::debug) { - auto pos = game::engine.getSystem<PlayerSystem>()->getPosition(); - ui::putText(offset.x - SCREEN_WIDTH2, (offset.y + SCREEN_HEIGHT2) - ui::fontSize, - "loc: %s\noffset: %s\nfps: %d\nticks: %d\npcount: %d\nxml: %s", - pos.toString().c_str(), offset.toString().c_str(), fps, - game::time::getTickCount(), game::engine.getSystem<ParticleSystem>()->getCount(), - game::engine.getSystem<WorldSystem>()->getXMLFile().c_str()); - } - - // draw the menu - if (currentMenu) - ui::menu::draw(); - - // draw the mouse - Render::textShader.use(); - glActiveTexture(GL_TEXTURE0); - mouseTex.use(); - Render::useShader(&Render::textShader); - Render::drawRect(vec2(ui::mouse.x, ui::mouse.y - 64), vec2(ui::mouse.x + 64, ui::mouse.y), -9.9); - Render::textShader.unuse(); -} |