#include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace std::literals::chrono_literals; std::string RenderSystem::loadTexString; Texture RenderSystem::loadTexResult; static std::vector randomDialog (readFileA("assets/dialog_en-us")); void MovementSystem::update(entityx::EntityManager &en, entityx::EventManager &ev, entityx::TimeDelta dt) { //bool fight = false; entityx::Entity toFight; (void)ev; en.each([&](entityx::Entity entity, Position &position, Direction &direction) { position.x += HLINES(direction.x) * dt; position.y += HLINES(direction.y) * dt; if (entity.has_component() && entity.has_component()) { auto animate = entity.component(); auto sprite = entity.component(); if (direction.x) animate->updateAnimation(1, sprite->sprite, dt); else animate->firstFrame(1, sprite->sprite); } if (entity.has_component() && entity.component()->talking) { direction.x = 0; } else { if (entity.has_component()) { auto& fl = entity.component()->faceLeft; if (direction.x != 0) fl = (direction.x < 0); } auto ppos = PlayerSystem::getPosition(); if (ppos.x > position.x && ppos.x < position.x + entity.component()->width) { if (entity.has_component()) { auto dim = entity.component(); ev.emit(vec2(position.x + dim->width, position.y + dim->height), AttackType::ShortSlash, false); /*auto& h = entity.component()->health; if (h > 0) { fight = true; toFight = entity; h = 0; }*/ } else if (entity.has_component()) { static bool triggering = false; if (!triggering) { triggering = true; std::thread([&](entityx::Entity e) { UISystem::fadeToggle(); UISystem::waitForCover(); UISystem::dialogImportant(e.component()->text); UISystem::waitForDialog(); UISystem::fadeToggle(); e.destroy(); triggering = false; }, entity).detach(); } return; } } // make the entity wander // TODO initialX and range? if (entity.has_component()) { auto& countdown = entity.component()->countdown; if (countdown > 0) { countdown--; } else { countdown = 5000 + randGet() % 10 * 100; direction.x = (randGet() % 3 - 1) * 0.004f; } } } }); // if (fight) { // UISystem::fadeToggleFast(); // UISystem::waitForCover(); //game::engine.getSystem()->fight(toFight); // UISystem::fadeToggleFast(); // } } void PhysicsSystem::update(entityx::EntityManager &en, entityx::EventManager &ev, entityx::TimeDelta dt) { (void)ev; en.each([dt](entityx::Entity entity, Direction &direction, Physics &physics) { (void)entity; // TODO GET GRAVITY FROM WORLD direction.y += physics.g * dt; }); } Texture RenderSystem::loadTexture(const std::string& file) { loadTexString = file; loadTexResult = Texture(); while (loadTexResult.isEmpty()) std::this_thread::sleep_for(1ms); auto t = loadTexResult; loadTexResult = Texture(); return t; } void RenderSystem::render(void) { if (!loadTexString.empty()) { loadTexResult = Texture(loadTexString, false); loadTexString.clear(); } if (!loadTexResult.isEmpty()) return; Render::worldShader.use(); Render::worldShader.enable(); game::entities.each([](entityx::Entity entity, Visible &visible, Sprite &sprite, Position &pos) { // Verticies and shit float its = 0; float sz; if (entity.has_component()) sz = entity.component()->width; else sz = sprite.getSpriteSize().x; if (sprite.faceLeft) { glm::mat4 scale = glm::scale(glm::mat4(1.0f), glm::vec3(-1.0f,1.0f,1.0f)); glm::mat4 translate = glm::translate(glm::mat4(1.0f), glm::vec3(0.0f - sz - pos.x * 2.0f, 0.0f, 0.0f)); glm::mat4 mov = scale * translate; glUniformMatrix4fv(Render::worldShader.uniform[WU_transform], 1, GL_FALSE, glm::value_ptr(mov)); } for (auto &S : sprite.sprite) { auto sp = S.first; auto size = sp.size * game::HLINE; vec2 drawOffset (HLINES(S.second.x), HLINES(S.second.y)); vec2 loc (pos.x + drawOffset.x, pos.y + drawOffset.y); GLfloat verts[] = { loc.x, loc.y, visible.z + its, sp.offset_tex.x, sp.offset_tex.y, loc.x + size.x, loc.y, visible.z + its, sp.offset_tex.x + sp.size_tex.x, sp.offset_tex.y, loc.x + size.x, loc.y + size.y, visible.z + its, sp.offset_tex.x + sp.size_tex.x, sp.offset_tex.y + sp.size_tex.y, loc.x, loc.y, visible.z + its, sp.offset_tex.x, sp.offset_tex.y, loc.x + size.x, loc.y + size.y, visible.z + its, sp.offset_tex.x + sp.size_tex.x, sp.offset_tex.y + sp.size_tex.y, loc.x, loc.y + size.y, visible.z + its, sp.offset_tex.x, sp.offset_tex.y + sp.size_tex.y }; // make the entity hit flash red // TODO /*if (maxHitDuration-hitDuration) { float flashAmt = 1-(hitDuration/maxHitDuration); glUniform4f(Render::worldShader.uniform[WU_tex_color], 1.0, flashAmt, flashAmt, 1.0); }*/ sp.tex.use(); glUniform1i(Render::worldShader.uniform[WU_texture], 0); glVertexAttribPointer(Render::worldShader.coord, 3, GL_FLOAT, GL_FALSE, 5 * sizeof(GLfloat), verts); glVertexAttribPointer(Render::worldShader.tex, 2, GL_FLOAT, GL_FALSE, 5 * sizeof(GLfloat), verts + 3); glDrawArrays(GL_TRIANGLES, 0, 6); //glUniform4f(Render::worldShader.uniform[WU_tex_color], 1.0, 1.0, 1.0, 1.0); its-=.01; } glUniformMatrix4fv(Render::worldShader.uniform[WU_transform], 1, GL_FALSE, glm::value_ptr(glm::mat4(1.0f))); if (entity.has_component()) { float width = entity.component()->width; auto& health = *entity.component(); width *= health.health / static_cast(health.maxHealth); GLfloat health_coord[] = { pos.x, pos.y, -9, 0, 0, pos.x + width, pos.y, -9, 0, 0, pos.x + width, pos.y - 5, -9, 0, 0, pos.x + width, pos.y - 5, -9, 0, 0, pos.x, pos.y - 5, -9, 0, 0, pos.x, pos.y, -9, 0, 0, }; Colors::red.use(); glVertexAttribPointer(Render::worldShader.coord, 3, GL_FLOAT, GL_FALSE, 5 * sizeof(GLfloat), health_coord); glVertexAttribPointer(Render::worldShader.tex, 2, GL_FLOAT, GL_FALSE, 5 * sizeof(GLfloat), health_coord + 3); glDrawArrays(GL_TRIANGLES, 0, 6); } }); Render::worldShader.disable(); Render::worldShader.unuse(); game::entities.each([](entityx::Entity e, Visible &v, Position &pos, Solid& dim, Name &name) { (void)e; (void)v; FontSystem::setFontZ(-5.0f); UISystem::putStringCentered(vec2(pos.x + dim.width / 2, pos.y - FontSystem::getSize() - HLINES(0.5)), name.name); }); } void DialogSystem::configure(entityx::EventManager &ev) { ev.subscribe(*this); } void DialogSystem::receive(const MouseClickEvent &mce) { game::entities.each( [&](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))) { if (!dialogRun.load()) { // copy entity, windows destroys the original after thread detach std::thread([e, &pos, &dim, &d, &name] { std::string questAssignedText; int newIndex; auto exml = WorldSystem::getXML()->FirstChildElement("Dialog"); dialogRun.store(true); if (e.has_component()) d.talking = true; if (d.index == 9999) { UISystem::dialogBox(name.name, /*"", false,*/ randomDialog[d.rindex % randomDialog.size()]); UISystem::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 ixml = exml->FirstChildElement("give"); if (ixml != nullptr) { do { InventorySystem::add(ixml->StrAttribute("name"), ixml->IntAttribute("count")); ixml = ixml->NextSiblingElement(); } while (ixml != nullptr); } auto qxml = exml->FirstChildElement("quest"); if (qxml != nullptr) { const char *qname; do { // assign quest qname = qxml->Attribute("assign"); if (qname != nullptr) { questAssignedText = qname; auto req = qxml->GetText(); QuestSystem::assign(qname, qxml->StrAttribute("desc"), req ? req : ""); } // check / finish quest else { qname = qxml->Attribute("check"); if (qname != nullptr) { if (qname != nullptr && QuestSystem::finish(qname) == 0) { d.index = 9999; } else { UISystem::dialogBox(name.name, /*"", false,*/ "Finish my quest u nug"); UISystem::waitForDialog(); return; } // oldidx = d.index; // d.index = qxml->UnsignedAttribute("fail"); // goto COMMONAIFUNC; } } } while((qxml = qxml->NextSiblingElement())); } auto xxml = exml->FirstChildElement("option"); std::string options; std::vector optionNexts; if (xxml != nullptr) { do { UISystem::dialogAddOption(xxml->StrAttribute("name")); options += '\"' + xxml->StrAttribute("name"); optionNexts.emplace_back(xxml->IntAttribute("value")); xxml = xxml->NextSiblingElement(); } while (xxml != nullptr); } 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)); } UISystem::dialogBox(name.name, /*options, false,*/ content); UISystem::waitForDialog(); UISystem::waitForDialog(); if (!questAssignedText.empty()) UISystem::dialogImportant("Quest assigned:\n\"" + questAssignedText + "\""); //passiveImportantText(5000, ("Quest assigned:\n\"" + questAssignedText + "\"").c_str()); if (exml->QueryIntAttribute("nextid", &newIndex) == XML_NO_ERROR) d.index = newIndex; } d.talking = false; dialogRun.store(false); }).detach(); } } }); } void DialogSystem::update(entityx::EntityManager &en, entityx::EventManager &ev, entityx::TimeDelta dt) { (void)en; (void)ev; (void)dt; } std::vector developFrame(XMLElement* xml) { Frame tmpf; std::vector tmp; SpriteData sd; unsigned int limb = 0; vec2 foffset; vec2 fsize; vec2 fdraw; tmp.clear(); // this is the xml elements first child. It will only be the tag auto framexml = xml->FirstChildElement(); while (framexml) { // this will always be frame. but if it isn't we don't wanna crash the game std::string defframe = framexml->Name(); if (defframe == "frame") { tmpf.clear(); // the xml element to parse each src of the frames auto sxml = framexml->FirstChildElement(); while (sxml) { std::string sname = sxml->Name(); if (sname == "src") { foffset = (sxml->Attribute("offset") != nullptr) ? sxml->StrAttribute("offset") : vec2(0,0); fdraw = (sxml->Attribute("drawOffset") != nullptr) ? sxml->StrAttribute("drawOffset") : vec2(0,0); if (sxml->Attribute("size") != nullptr) { fsize = sxml->StrAttribute("size"); sd = SpriteData(sxml->GetText(), foffset, fsize); } else { sd = SpriteData(sxml->GetText(), foffset); } if (sxml->QueryUnsignedAttribute("limb", &limb) == XML_NO_ERROR) sd.limb = limb; tmpf.push_back(std::make_pair(sd, fdraw)); } sxml = sxml->NextSiblingElement(); } // we don't want segfault if (tmpf.size()) tmp.push_back(tmpf); } // if it's not a frame we don't care // parse next frame framexml = framexml->NextSiblingElement(); } return tmp; }