diff options
-rw-r--r-- | include/quest.hpp | 72 | ||||
-rw-r--r-- | main.cpp | 5 | ||||
-rw-r--r-- | src/components.cpp | 36 | ||||
-rw-r--r-- | src/engine.cpp | 7 | ||||
-rw-r--r-- | src/quest.cpp | 46 | ||||
-rw-r--r-- | src/ui.cpp | 17 | ||||
-rw-r--r-- | src/ui_menu.cpp | 10 |
7 files changed, 178 insertions, 15 deletions
diff --git a/include/quest.hpp b/include/quest.hpp new file mode 100644 index 0000000..cb43eca --- /dev/null +++ b/include/quest.hpp @@ -0,0 +1,72 @@ +/** + * @file quest.hpp + * Quest handling. + */ + +#ifndef QUEST_HPP_ +#define QUEST_HPP_ + +#include <entityx/entityx.h> + +#include <string> +#include <vector> + +/** + * The Quest structure. + * Contains information necessary for a quest. + */ +struct Quest +{ + Quest(std::string n = "", std::string d = "") + : name(n), desc(d) {} + + std::string name; /**< the quest's title */ + std::string desc; /**< the quest's description */ +}; + +/** + * @class QuestSystem + * The quest system, handles active quests. + */ +class QuestSystem : public entityx::System<QuestSystem> { +private: + /** + * A list of all quests that are currently active. + */ + std::vector<Quest> current; + +public: + void update(entityx::EntityManager &en, entityx::EventManager &ev, entityx::TimeDelta dt) override; + + /** + * Adds a quest to the current quest vector by its title. + * @param title the quest's title + * @param desc the quest's description + * @param req retrieved from XML, list of what the quest wants + * @return a possible error code + */ + int assign(std::string title, std::string desc, std::string req); + + /** + * Drops a quest through its title. + * @param title the quest's title + * @return a possible error code + */ + int drop(std::string title); + + /** + * Finishes a quest through it's title. + * @param title the quest's title + * @return a possible error code + */ + int finish(std::string title); + + /** + * Returns true if the system is currently taking the quest. + * @param title the quest's title + * @return if the quest is active. + */ + bool hasQuest(std::string title); +}; + +#endif // QUEST_HPP_ @@ -259,7 +259,8 @@ void render() { const auto SCREEN_HEIGHT = game::SCREEN_HEIGHT; auto ps = game::engine.getSystem<PlayerSystem>(); - offset.x = ps->getPosition().x + ps->getWidth() / 2; + auto ploc = ps->getPosition(); + offset.x = ploc.x + ps->getWidth() / 2; const auto& worldWidth = game::engine.getSystem<WorldSystem>()->getWidth(); if (worldWidth < (int)SCREEN_WIDTH) @@ -270,7 +271,7 @@ void render() { offset.x = ((worldWidth * 0.5f) - SCREEN_WIDTH / 2); // ortho y snapping - offset.y = /*std::max(player->loc.y + player->height / 2,*/ SCREEN_HEIGHT / 2.0f; /*);*/ + offset.y = std::max(ploc.y /*+ player->height / 2*/, SCREEN_HEIGHT / 2.0f); // "setup" glm::mat4 projection = glm::ortho(floor(offset.x - SCREEN_WIDTH / 2), // left diff --git a/src/components.cpp b/src/components.cpp index 38fd7a6..eb9fb0e 100644 --- a/src/components.cpp +++ b/src/components.cpp @@ -8,6 +8,7 @@ #include <engine.hpp> #include <world.hpp> #include <brice.hpp> +#include <quest.hpp> static std::vector<std::string> randomDialog (readFileA("assets/dialog_en-us")); @@ -158,9 +159,11 @@ void DialogSystem::receive(const MouseClickEvent &mce) ((mce.position.y > pos.y) & (mce.position.y < pos.y + dim.height))) { std::thread([&] { - auto exml = game::engine.getSystem<WorldSystem>()->getXML()->FirstChildElement("Dialog"); + std::string questAssignedText; int newIndex; + auto exml = game::engine.getSystem<WorldSystem>()->getXML()->FirstChildElement("Dialog"); + if (e.has_component<Direction>()) d.talking = true; @@ -182,6 +185,34 @@ void DialogSystem::receive(const MouseClickEvent &mce) game::briceUpdate(); } + auto qxml = exml->FirstChildElement("quest"); + if (qxml != nullptr) { + std::string qname; + auto qsys = game::engine.getSystem<QuestSystem>(); + + do { + // assign quest + qname = qxml->StrAttribute("assign"); + if (!qname.empty()) { + questAssignedText = qname; + qsys->assign(qname, qxml->StrAttribute("desc"), "req"); // gettext() for req + } + + // check / finish quest + else { + qname = qxml->StrAttribute("check"); + if (!(qname.empty() && qsys->hasQuest(qname) && qsys->finish(qname))) { + 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) { @@ -194,6 +225,9 @@ void DialogSystem::receive(const MouseClickEvent &mce) ui::dialogBox(name.name, "", false, content); ui::waitForDialog(); + if (!questAssignedText.empty()) + ui::passiveImportantText(4000, ("Quest assigned:\n\"" + questAssignedText + "\"").c_str()); + if (exml->QueryIntAttribute("nextid", &newIndex) == XML_NO_ERROR) d.index = newIndex; } diff --git a/src/engine.cpp b/src/engine.cpp index 7c9a404..e5b2b36 100644 --- a/src/engine.cpp +++ b/src/engine.cpp @@ -7,6 +7,7 @@ #include <inventory.hpp> #include <components.hpp> #include <player.hpp> +#include <quest.hpp> Engine::Engine(void) : shouldRun(true), systems(game::entities, game::events) @@ -23,6 +24,7 @@ void Engine::init(void) { //systems.add<InventorySystem>(); systems.add<WorldSystem>(); systems.add<PlayerSystem>(); + systems.add<QuestSystem>(); systems.add<PhysicsSystem>(); systems.add<MovementSystem>(); @@ -39,7 +41,7 @@ void Engine::render(entityx::TimeDelta dt) { systems.update<RenderSystem>(dt); systems.update<WindowSystem>(dt); - //systems.update<InventorySystem>(dt); + //systems.update<InventorySystem>(dt); // doesn't do anything... ui::fadeUpdate(); } @@ -48,9 +50,10 @@ void Engine::update(entityx::TimeDelta dt) { systems.update<InputSystem>(dt); systems.update<MovementSystem>(dt); - systems.update<DialogSystem>(dt); + //systems.update<DialogSystem>(dt); // doesn't do anything systems.update<WorldSystem>(dt); systems.update<PlayerSystem>(dt); + //systems.update<QuestSystem>(dt); // doesn't do anything } diff --git a/src/quest.cpp b/src/quest.cpp new file mode 100644 index 0000000..f7548d2 --- /dev/null +++ b/src/quest.cpp @@ -0,0 +1,46 @@ +#include <quest.hpp> + +#include <algorithm> + +void QuestSystem::update(entityx::EntityManager &en, entityx::EventManager &ev, entityx::TimeDelta dt) +{ + (void)en; + (void)ev; + (void)dt; +} + +int QuestSystem::assign(std::string title, std::string desc, std::string req) +{ + (void)req; + current.emplace_back(title, desc); + return 0; +} + +int QuestSystem::drop(std::string title) +{ + current.erase(std::remove_if(std::begin(current), std::end(current), + [&title](const Quest& q) { return (q.name == title); })); + return 0; +} + +int QuestSystem::finish(std::string title) +{ + auto quest = std::find_if(std::begin(current), std::end(current), + [&title](const Quest& q) { return (q.name == title); }); + + if (quest == std::end(current)) + return -1; + + // TODO requirements + + drop(title); + + return 0; +} + +bool QuestSystem::hasQuest(std::string title) +{ + return (std::find_if(std::begin(current), std::end(current), + [&title](const Quest& q) { return (q.name == title); }) != std::end(current)); +} + @@ -411,13 +411,16 @@ namespace ui { } float putStringCentered(const float x, const float y, std::string s) { - unsigned int i = 0; - float width = 0; + unsigned int i = 0, lastnl = 0; + float width = 0, yy = y; do { switch (s[i]) { case '\n': - // TODO + putString(floor(x - width / 2), yy, s.substr(0, i)); + lastnl = 1 + i; + width = 0; + yy -= fontSize * 1.15f; break; case '\b': break; @@ -429,7 +432,8 @@ namespace ui { break; } } while(s[++i]); - putString(floor(x-width/2),y,s); + + putString(floor(x - width / 2), yy, s.substr(lastnl)); return width; } @@ -478,17 +482,20 @@ namespace ui { case '!': case '?': case '.': + case ':': tadv = 10; break; case ',': + case ';': tadv = 5; break; default: tadv = 1; break; } - } else + } else { typeOutDone = true; + } } return ret; // The buffered string. diff --git a/src/ui_menu.cpp b/src/ui_menu.cpp index f6f962d..59b44e6 100644 --- a/src/ui_menu.cpp +++ b/src/ui_menu.cpp @@ -159,11 +159,11 @@ namespace ui { void init(void) { // Create the main pause menu - pauseMenu.items.push_back(ui::menu::createParentButton({-128,0},{256,75},{0.0f,0.0f,0.0f}, "Resume")); - pauseMenu.items.push_back(ui::menu::createChildButton({-128,-100},{256,75},{0.0f,0.0f,0.0f}, "Options", &optionsMenu)); - pauseMenu.items.push_back(ui::menu::createChildButton({-128,-200},{256,75},{0.0f,0.0f,0.0f}, "Controls", &controlsMenu)); - pauseMenu.items.push_back(ui::menu::createButton({-128,-300},{256,75},{0.0f,0.0f,0.0f}, "Save and Quit", ui::quitGame)); - pauseMenu.items.push_back(ui::menu::createButton({-128,-400},{256,75},{0.0f,0.0f,0.0f}, "Segfault", segFault)); + pauseMenu.items.push_back(ui::menu::createParentButton({-128,100},{256,75},{0.0f,0.0f,0.0f}, "Resume")); + pauseMenu.items.push_back(ui::menu::createChildButton({-128, 0},{256,75},{0.0f,0.0f,0.0f}, "Options", &optionsMenu)); + pauseMenu.items.push_back(ui::menu::createChildButton({-128,-100},{256,75},{0.0f,0.0f,0.0f}, "Controls", &controlsMenu)); + pauseMenu.items.push_back(ui::menu::createButton({-128,-200},{256,75},{0.0f,0.0f,0.0f}, "Save and Quit", ui::quitGame)); + pauseMenu.items.push_back(ui::menu::createButton({-128,-300},{256,75},{0.0f,0.0f,0.0f}, "Segfault", segFault)); // Create the options (sound) menu optionsMenu.items.push_back(ui::menu::createSlider({0-static_cast<float>(game::SCREEN_WIDTH)/4,0-(512/2)}, {50,512}, {0.0f, 0.0f, 0.0f}, 0, 100, "Master", &game::config::VOLUME_MASTER)); |