diff options
-rw-r--r-- | include/brice.hpp | 39 | ||||
-rw-r--r-- | include/components.hpp | 2 | ||||
-rw-r--r-- | include/config.hpp | 49 | ||||
-rw-r--r-- | include/engine.hpp | 63 | ||||
-rw-r--r-- | include/gametime.hpp | 36 | ||||
-rw-r--r-- | include/save_util.hpp | 36 | ||||
-rw-r--r-- | include/shader_utils.hpp | 3 | ||||
-rw-r--r-- | include/ui_quest.hpp | 19 | ||||
-rw-r--r-- | src/components.cpp | 172 | ||||
-rw-r--r-- | src/ui.cpp | 8 | ||||
-rw-r--r-- | src/ui_menu.cpp | 6 | ||||
-rw-r--r-- | src/world.cpp | 16 |
12 files changed, 302 insertions, 147 deletions
diff --git a/include/brice.hpp b/include/brice.hpp index dc3ea96..1c4eccf 100644 --- a/include/brice.hpp +++ b/include/brice.hpp @@ -1,20 +1,59 @@ +/** + * @file brice.hpp + * @brief A system for saving player information. + */ + #ifndef BRICE_H_ #define BRICE_H_ #include <string> namespace game { + + /** + * Allows the player to jump, if set to true. + */ extern bool canJump; + + /** + * Allows the player to sprint, if set to true. + */ extern bool canSprint; + /** + * Gets a value from the saved brice and returns it. + * @param id the id of the value + * @return the string value + */ std::string getValue(const std::string& id); + /** + * Sets a value in the brice, creating it if it doesn't exist. + * @param id the id of the value + * @param value the value + * @return true if the value was updated, not created + */ bool setValue(const std::string& id, const std::string& value); + /** + * Resets the brice to it's default values. + * Note: these are hardcoded into the program. + */ void briceClear(void); + + /** + * Saves the brice to it's file (brice.dat). + */ void briceSave(void); + + /** + * Loads the brice from it's file (brice.dat). + */ void briceLoad(void); + /** + * Reloads the brice. + */ void briceUpdate(void); } diff --git a/include/components.hpp b/include/components.hpp index d630f83..2e09fc7 100644 --- a/include/components.hpp +++ b/include/components.hpp @@ -151,7 +151,7 @@ struct Sprite { Sprite(bool left = false) : faceLeft(left) {} - std::vector<std::pair<SpriteData, vec2>> getSprite() { + Frame getSprite() { return sprite; } diff --git a/include/config.hpp b/include/config.hpp index bc9d052..908c376 100644 --- a/include/config.hpp +++ b/include/config.hpp @@ -1,23 +1,72 @@ +/** + * @file config.hpp + * @brief Functions for loading/saving game settings. + */ + #ifndef CONFIG_H #define CONFIG_H #include <string> namespace game { + /** + * The size of an HLINE, according to the save file. + * This is the default "unit of measurement" in the game. Drawing scales to + * this, and it is used in game logic. + */ extern unsigned int HLINE; + + /** + * The width of the screen, in pixels. + */ extern unsigned int SCREEN_WIDTH; + + /** + * The height of the screen, in pixels. + */ extern unsigned int SCREEN_HEIGHT; + + /** + * The window is fullscreen if this is true. + */ extern bool FULLSCREEN; namespace config { + /** + * The current volume level of the master channel. + * Volumes are percentages, 0 to 100. + */ extern float VOLUME_MASTER; + + /** + * Volume level of the background music (BGM). + */ extern float VOLUME_MUSIC; + + /** + * Volume level of game sound effects. + */ extern float VOLUME_SFX; + /** + * The path of the folder to load world XML files from. + */ extern std::string xmlFolder; + /** + * Reads the settings file (config/settings.xml) into the game. + * Default values are hardcoded in (see src/config.cpp). + */ void read(void); + + /** + * Updates settings with the current values. + */ void update(void); + + /** + * Saves the current settings to the settings file. + */ void save(void); } } diff --git a/include/engine.hpp b/include/engine.hpp index c842b59..d4028d8 100644 --- a/include/engine.hpp +++ b/include/engine.hpp @@ -1,8 +1,13 @@ +/** + * @file engine.hpp + * @brief The main game engine, and functions to assist it. + */ + #ifndef ENGINE_HPP_ #define ENGINE_HPP_ #include <entityx/entityx.h> -#include "entityx/deps/Dependencies.h" +#include <entityx/deps/Dependencies.h> #include <texture.hpp> #include <components.hpp> @@ -10,27 +15,55 @@ //game::engine::Systems->add<entityx::deps::Dependency<Visible, Sprite>>(); +/** + * @class Engine + * The main game engine class. Only one instance of this should be created, it + * handles everything game-related. + */ class Engine : public entityx::Receiver<Engine> { public: + /** + * A flag to indicate if a thread should continue to run. + */ bool shouldRun; + /** + * Handles game systems. + */ entityx::SystemManager systems; explicit Engine(void); + /** + * Initializes the game engine, and all systems used within it. + */ void init(void); + + /** + * Updates all rendering systems. + * @param dt the delta time + */ void render(entityx::TimeDelta dt); + + /** + * Updates all logic systems. + * @param dt the delta time + */ void update(entityx::TimeDelta dt); + /** + * A shortcut to get a system, for calling system-specific functions. + * Takes the type of the desired system. + */ template<typename T> inline T* getSystem(void) { return dynamic_cast<T*>(systems.system<T>().get()); } - /*void configure(entityx::EventManager &ev) { - (void)ev; - }*/ - + /** + * A handler for the game ending event. + * @param gee game end event data + */ inline void receive(const GameEndEvent &gee) { shouldRun = !(gee.really); } @@ -38,16 +71,32 @@ public: namespace game { + /** + * Handles all game events. + */ extern entityx::EventManager events; + + /** + * Handles entity data. + */ extern entityx::EntityManager entities; + /** + * Handles sprite loading, for the sprite system. + */ + extern SpriteLoader sprite_l; + + /** + * An instance of the main game engine. + */ extern Engine engine; + /** + * Ends the game. + */ inline void endGame(void) { events.emit<GameEndEvent>(); } - - extern SpriteLoader sprite_l; } diff --git a/include/gametime.hpp b/include/gametime.hpp index a809ef9..988533a 100644 --- a/include/gametime.hpp +++ b/include/gametime.hpp @@ -1,16 +1,52 @@ +/** + * @file gametime.hpp + * @brief Handles time related operations + */ + #ifndef GAMETIME_H_ #define GAMETIME_H_ namespace game { namespace time { + /** + * Sets the game's tick count to the desired amount. + * @param t desired tick count + */ void setTickCount(unsigned int t); + + /** + * Gets the current tick count. + * @return the tick count + */ unsigned int getTickCount(void); + + /** + * Calculates and returns the delta time. + * @return the delta time + */ unsigned int getDeltaTime(void); + /** + * Increments the game's tick count. + */ void tick(void); + + /** + * Increments the game's tick count by the given amount of ticks. + * @param ticks the number of ticks to add + */ void tick(unsigned int ticks); + + /** + * Determines if a tick has passed since the last call to this function. + * @return if a tick has passed + */ bool tickHasPassed(void); + /** + * Handles time updating. + * This should be called from the game's main loop. + */ void mainLoopHandler(void); } } diff --git a/include/save_util.hpp b/include/save_util.hpp deleted file mode 100644 index 3d5cf54..0000000 --- a/include/save_util.hpp +++ /dev/null @@ -1,36 +0,0 @@ -#ifndef SAVE_UTIL_H_ -#define SAVE_UTIL_H_ - -/* - * Save macros. - */ - -#define E_SAVE_COORDS { xmle->SetAttribute("x", loc.x); xmle->SetAttribute("y", loc.y); } - -#define E_SAVE_HEALTH xmle->SetAttribute("health", health); - -/* - * Load macos. - */ - -#define E_LOAD_COORDS(yy) { float n; \ - if (xmle->QueryFloatAttribute("x", &n) == XML_NO_ERROR) \ - spawn(n, yy); \ - else \ - spawn(xmle->FloatAttribute("spawnx"), 100); \ - \ - if (xmle->QueryFloatAttribute("y", &n) == XML_NO_ERROR) \ - loc.y = n; } - -#define E_LOAD_HEALTH { float n; \ - \ - if (xmle->QueryFloatAttribute("maxHealth", &n) != XML_NO_ERROR) \ - maxHealth = 1; \ - \ - if (xmle->QueryFloatAttribute("health", &n) == XML_NO_ERROR) \ - health = n; \ - else \ - health = maxHealth; } - - -#endif // SAVE_UTIL_H_ diff --git a/include/shader_utils.hpp b/include/shader_utils.hpp index 243b3d4..08ca7b3 100644 --- a/include/shader_utils.hpp +++ b/include/shader_utils.hpp @@ -1,8 +1,11 @@ /** + * @file shader_utils.hpp + * @brief Utilities to use to handle GLSL shaders. * From the OpenGL Programming wikibook: http://en.wikibooks.org/wiki/OpenGL_Programming * This file is in the public domain. * Contributors: Sylvain Beucler, Guus Sliepen */ + #ifndef _CREATE_SHADER_H #define _CREATE_SHADER_H diff --git a/include/ui_quest.hpp b/include/ui_quest.hpp index 8582b67..d770011 100644 --- a/include/ui_quest.hpp +++ b/include/ui_quest.hpp @@ -1,3 +1,8 @@ +/** + * @file ui_quest.hpp + * @brief Handles UI elements related to quests. + */ + #ifndef UI_QUEST_HPP_ #define UI_QUEST_HPP_ @@ -6,12 +11,20 @@ namespace ui { namespace quest { + /** + * A flag to determine if the UI should be drawn. + */ bool _toggle = false; - void toggle(void) { - _toggle ^= true; - } + /** + * Toggles displaying of the UI. + */ + inline void toggle(void) + { _toggle ^= true; } + /** + * Draws the quest UI to the screen, if enabled. + */ void draw(void) { static unsigned int textWrap = 40; diff --git a/src/components.cpp b/src/components.cpp index 612e522..e85883e 100644 --- a/src/components.cpp +++ b/src/components.cpp @@ -10,6 +10,8 @@ #include <brice.hpp> #include <quest.hpp> +#include <atomic> + static std::vector<std::string> randomDialog (readFileA("assets/dialog_en-us")); void MovementSystem::update(entityx::EntityManager &en, entityx::EventManager &ev, entityx::TimeDelta dt) @@ -20,11 +22,9 @@ void MovementSystem::update(entityx::EntityManager &en, entityx::EventManager &e position.y += direction.y * dt; if (entity.has_component<Animate>() && entity.has_component<Sprite>()) { - if (direction.x) { - entity.component<Sprite>().get()->sprite = entity.component<Animate>().get()->nextFrame(); - } else { - entity.component<Sprite>().get()->sprite = entity.component<Animate>().get()->firstFrame(); - } + auto animate = entity.component<Animate>(); + entity.component<Sprite>()->sprite = + (direction.x != 0) ? animate->nextFrame() : animate->firstFrame(); } if (entity.has_component<Dialog>() && entity.component<Dialog>()->talking) { direction.x = 0; @@ -98,9 +98,6 @@ void RenderSystem::update(entityx::EntityManager &en, entityx::EventManager &ev, 1.0, 1.0, 1.0, 0.0}; - if (entity.has_component<Animate>()) - sprite.sprite = entity.component<Animate>()->nextFrame(); - for (auto &S : sprite.sprite) { float width = HLINES(S.first.size.x); float height = HLINES(S.first.size.y); @@ -159,97 +156,100 @@ void DialogSystem::receive(const MouseClickEvent &mce) { game::entities.each<Position, Solid, Dialog, Name>( [&](entityx::Entity e, Position &pos, Solid &dim, Dialog &d, Name &name) { + static std::atomic_bool dialogRun; (void)e; (void)d; if (((mce.position.x > pos.x) & (mce.position.x < pos.x + dim.width)) && ((mce.position.y > pos.y) & (mce.position.y < pos.y + dim.height))) { - std::thread([&] { - std::string questAssignedText; - int newIndex; - - auto exml = game::engine.getSystem<WorldSystem>()->getXML()->FirstChildElement("Dialog"); - - if (e.has_component<Direction>()) - d.talking = true; - - if (d.index == 9999) { - ui::dialogBox(name.name, "", false, randomDialog[d.rindex % randomDialog.size()]); - ui::waitForDialog(); - } else if (exml != nullptr) { - while (exml->StrAttribute("name") != name.name) - exml = exml->NextSiblingElement(); - - exml = exml->FirstChildElement("text"); - while (exml->IntAttribute("id") != d.index) - exml = exml->NextSiblingElement(); - - auto oxml = exml->FirstChildElement("set"); - if (oxml != nullptr) { - do game::setValue(oxml->StrAttribute("id"), oxml->StrAttribute("value")); - while ((oxml = oxml->NextSiblingElement())); - game::briceUpdate(); - } - - auto qxml = exml->FirstChildElement("quest"); - if (qxml != nullptr) { - const char *qname; - auto qsys = game::engine.getSystem<QuestSystem>(); - - do { - // assign quest - qname = qxml->Attribute("assign"); - if (qname != nullptr) { - questAssignedText = qname; - auto req = qxml->GetText(); - qsys->assign(qname, qxml->StrAttribute("desc"), req ? req : ""); - } - - // check / finish quest - else { - qname = qxml->Attribute("check"); + if (!dialogRun.load()) { + std::thread([&] { + std::string questAssignedText; + int newIndex; + + auto exml = game::engine.getSystem<WorldSystem>()->getXML()->FirstChildElement("Dialog"); + dialogRun.store(true); + + if (e.has_component<Direction>()) + d.talking = true; + + if (d.index == 9999) { + ui::dialogBox(name.name, "", false, randomDialog[d.rindex % randomDialog.size()]); + ui::waitForDialog(); + } else if (exml != nullptr) { + while (exml->StrAttribute("name") != name.name) + exml = exml->NextSiblingElement(); + + exml = exml->FirstChildElement("text"); + while (exml->IntAttribute("id") != d.index) + exml = exml->NextSiblingElement(); + + auto oxml = exml->FirstChildElement("set"); + if (oxml != nullptr) { + do game::setValue(oxml->StrAttribute("id"), oxml->StrAttribute("value")); + while ((oxml = oxml->NextSiblingElement())); + game::briceUpdate(); + } + + auto qxml = exml->FirstChildElement("quest"); + if (qxml != nullptr) { + const char *qname; + auto qsys = game::engine.getSystem<QuestSystem>(); + + do { + // assign quest + qname = qxml->Attribute("assign"); if (qname != nullptr) { - if (qname != nullptr && qsys->hasQuest(qname) && qsys->finish(qname) == 0) { - d.index = 9999; - } else { - ui::dialogBox(name.name, "", false, "Finish my quest u nug"); - ui::waitForDialog(); - return; - } - // oldidx = d.index; - // d.index = qxml->UnsignedAttribute("fail"); - // goto COMMONAIFUNC; + questAssignedText = qname; + auto req = qxml->GetText(); + qsys->assign(qname, qxml->StrAttribute("desc"), req ? req : ""); } - } - } while((qxml = qxml->NextSiblingElement())); - } - auto cxml = exml->FirstChildElement("content"); - const char *content; - if (cxml == nullptr) { - content = randomDialog[d.rindex % randomDialog.size()].c_str(); - } else { - content = cxml->GetText() - 1; - while (*++content && isspace(*content)); + // check / finish quest + else { + qname = qxml->Attribute("check"); + if (qname != nullptr) { + if (qname != nullptr && qsys->hasQuest(qname) && qsys->finish(qname) == 0) { + d.index = 9999; + } else { + ui::dialogBox(name.name, "", false, "Finish my quest u nug"); + ui::waitForDialog(); + return; + } + // oldidx = d.index; + // d.index = qxml->UnsignedAttribute("fail"); + // goto COMMONAIFUNC; + } + } + } while((qxml = qxml->NextSiblingElement())); + } + + auto cxml = exml->FirstChildElement("content"); + const char *content; + if (cxml == nullptr) { + content = randomDialog[d.rindex % randomDialog.size()].c_str(); + } else { + content = cxml->GetText() - 1; + while (*++content && isspace(*content)); + } + + ui::dialogBox(name.name, "", false, content); + ui::waitForDialog(); + + if (!questAssignedText.empty()) + ui::passiveImportantText(5000, ("Quest assigned:\n\"" + questAssignedText + "\"").c_str()); + + if (exml->QueryIntAttribute("nextid", &newIndex) == XML_NO_ERROR) + d.index = newIndex; } - ui::dialogBox(name.name, "", false, content); - ui::waitForDialog(); - - if (!questAssignedText.empty()) - ui::passiveImportantText(5000, ("Quest assigned:\n\"" + questAssignedText + "\"").c_str()); - - if (exml->QueryIntAttribute("nextid", &newIndex) == XML_NO_ERROR) - d.index = newIndex; - } - - d.talking = false; - }).detach(); - + d.talking = false; + dialogRun.store(false); + }).detach(); } } - ); + }); } void DialogSystem::update(entityx::EntityManager &en, entityx::EventManager &ev, entityx::TimeDelta dt) @@ -200,7 +200,7 @@ namespace ui { UserError("Couldn't initialize freetype."); #ifdef DEBUG - DEBUG_printf("Initialized FreeType2.\n",NULL); + DEBUG_printf("Initialized FreeType2.\n", nullptr); #endif // DEBUG fontSize = 0; @@ -562,7 +562,7 @@ namespace ui { // cycle through options while (sopt) { dialogOptText.push_back(std::make_pair((std::string)sopt, vec3 {0,0,0})); - sopt = strtok(NULL,":"); + sopt = strtok(nullptr, ":"); } } @@ -1037,7 +1037,7 @@ namespace ui { void quitGame() { dialogBoxExists = false; - currentMenu = NULL; + currentMenu = nullptr; game::config::update(); game::config::save(); game::endGame(); @@ -1183,7 +1183,7 @@ EXIT: bgr[x+2] = pixels[x]; } - time_t epoch = time(NULL); + time_t epoch = time(nullptr); struct tm* timen = localtime(&epoch); std::string name = "screenshots/"; diff --git a/src/ui_menu.cpp b/src/ui_menu.cpp index 59b44e6..d853abc 100644 --- a/src/ui_menu.cpp +++ b/src/ui_menu.cpp @@ -120,7 +120,7 @@ namespace ui { temp.button.dim = d; temp.button.color = c; temp.button.text = t; - temp.button.func = NULL; + temp.button.func = nullptr; temp.child = _child; return temp; @@ -134,7 +134,7 @@ namespace ui { temp.button.dim = d; temp.button.color = c; temp.button.text = t; - temp.button.func = NULL; + temp.button.func = nullptr; temp.child = nullptr; return temp; @@ -320,7 +320,7 @@ namespace ui { cMult = 0.75f; //if we are inside the slider and click it will set the slider to that point - if (SDL_GetMouseState(NULL, NULL) & SDL_BUTTON(SDL_BUTTON_LEFT)) { + if (SDL_GetMouseState(nullptr, nullptr) & SDL_BUTTON(SDL_BUTTON_LEFT)) { //change handle location if (m.slider.dim.y > m.slider.dim.x) { *m.slider.var = (((mouse.y-offset.y) - m.slider.loc.y)/m.slider.dim.y)*100; diff --git a/src/world.cpp b/src/world.cpp index 4a7e284..ea2c670 100644 --- a/src/world.cpp +++ b/src/world.cpp @@ -190,17 +190,14 @@ void WorldSystem::load(const std::string& file) entityx::Entity entity; - std::string xmlRaw; - std::string xmlPath; - // check for empty file name if (file.empty()) return; // load file data to string - xmlPath = xmlFolder + file; + auto xmlPath = xmlFolder + file; auto xmlRawData = readFile(xmlPath.c_str()); - xmlRaw = xmlRawData; + std::string xmlRaw = xmlRawData; delete[] xmlRawData; // let tinyxml parse the file @@ -209,18 +206,23 @@ void WorldSystem::load(const std::string& file) // include headers auto ixml = xmlDoc.FirstChildElement("include"); - while (ixml) { + while (ixml != nullptr) { auto file = ixml->Attribute("file"); + if (file != nullptr) { DEBUG_printf("Including file: %s\n", file); auto include = readFile((xmlFolder + file).c_str()); xmlRaw.append(include); delete[] include; + } else { + UserError("XML Error: <include> tag file not given"); } + ixml = ixml->NextSiblingElement(); } - xmlDoc.Parse(xmlRaw.data()); + if (xmlDoc.Parse(xmlRaw.data()) != XML_NO_ERROR) + UserError("XML Error:"); // look for an opening world tag auto wxml = xmlDoc.FirstChildElement("World"); |