#include #include #include #include #include #include #include // math helpers because we don't trust stdlib template inline T abs(const T& n) { static_assert(std::is_arithmetic::value, "abs expects numbers"); return n >= 0 ? n : -n; } bool inrange(float point, float left, float right, float range) { return (left < point + range && left > point - range) || (right < point + range && right > point - range) || (point > left && point < right); } bool inrange(float point, float left, float right) { return point > left && point < right; } std::vector AttackSystem::effects; bool AttackSystem::receive(const AttackEvent& ae) { attacks.emplace_front(ae); return true; } namespace lua { static entityx::Entity* entity; inline void setEntity(entityx::Entity* e) { entity = e; } int flash(lua_State* state) { float r = lua_tonumber(state, 1); float g = lua_tonumber(state, 2); float b = lua_tonumber(state, 3); entity->replace(Color(r, g, b)); return 0; } int damage(lua_State* state) { float d = lua_tonumber(state, 1); entity->component()->health -= d; return 0; } } void AttackSystem::initLua(LuaScript& s) { s.addFunction("flash", lua::flash); s.addFunction("damage", lua::damage); } void AttackSystem::update(entityx::EntityManager& en, entityx::EventManager& ev, entityx::TimeDelta dt) { (void)en; (void)ev; (void)dt; // handle painful entities (e.g. arrow) en.each([&](entityx::Entity p, Hit& hit, Position& ppos) { bool die = false; en.each([&](entityx::Entity e, Health& health, Position& pos, Solid& dim) { (void)health; if (!e.has_component() && inrange(ppos.x, pos.x, pos.x + dim.width) && inrange(ppos.y, pos.y - 2, pos.y + dim.height)) { lua::setEntity(&e); hit.attack->script("effect"); if (hit.effect.size() > 0) effects.emplace_back(vec2(ppos.x, ppos.y), hit.effect); //ParticleSystem::addMultiple(15, ParticleType::SmallBlast, // [&](){ return vec2(pos.x + dim.width / 2, pos.y + dim.height / 2); }, 300, 7); die = true; } else if (WorldSystem::isAboveGround(vec2(ppos.x, ppos.y - 5))) die = true; }); if (die) p.destroy(); }); // handle emitted attacks (player's) for (const auto& a : attacks) { //vec2 point = a.pos + a.attack.offset; //vec2 size = a.attack.range; //point.y -= size.y / 2; // center range height vec2 point = a.pos + a.attack.offset; vec2 size (0, a.attack.range.y); point.y -= size.y / 2; a.attack.script("hit", {LuaVariable("xrange", size.x)}); en.each( [&](entityx::Entity e, Position& pos, Solid& dim, Health& h) { (void)h; if (!(e.has_component() ^ a.fromplayer)) // no self-harm please return; if (inrange(point.x, pos.x, pos.x + dim.width, HLINES(size.x)) && inrange(point.y, pos.y, pos.y + dim.height, HLINES(size.y))) { lua::setEntity(&e); a.attack.script("effect"); if (pos.x < point.x) { e.component()->x = -0.1; e.component()->y = 0.1; } else { e.component()->x = 0.1; e.component()->y = 0.1; } if (a.attack.effect.size() > 0) effects.emplace_back(point, a.attack.effect); //ParticleSystem::addMultiple(15, ParticleType::DownSlash, // [&](){ return vec2(pos.x + dim.width / 2, pos.y + dim.height / 2); }, 300, 7); } } ); } attacks.clear(); } #define RATE 3 void AttackSystem::render(void) { float z = Render::ZRange::Attack; Render::worldShader.use(); Render::worldShader.enable(); for (auto& ae : effects) { ae.effect(ae.counter / RATE); // bind current frame auto dim = ae.effect.getTextureDim(); auto s = ae.pos - (dim / 2); GLfloat verts[] = { s.x, s.y, z, 0, 0, s.x + dim.x, s.y, z, 1, 0, s.x + dim.x, s.y + dim.y, z, 1, 1, s.x + dim.x, s.y + dim.y, z, 1, 1, s.x, s.y + dim.y, z, 0, 1, s.x, s.y, z, 0, 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); z -= 0.05f; } Render::worldShader.disable(); Render::worldShader.unuse(); effects.erase(std::remove_if(effects.begin(), effects.end(), [](auto& e) { return ++e.counter >= e.effect.size() * RATE; }), effects.end()); }