diff options
39 files changed, 843 insertions, 432 deletions
diff --git a/Assets/world/world1/layers/0/forestTileBack.png b/Assets/world/world1/layers/0/forestTileBack.png Binary files differnew file mode 100644 index 0000000..f1c1a8b --- /dev/null +++ b/Assets/world/world1/layers/0/forestTileBack.png diff --git a/Assets/world/world1/layers/0/hitbox.png b/Assets/world/world1/layers/0/hitbox.png Binary files differnew file mode 100644 index 0000000..9ae9622 --- /dev/null +++ b/Assets/world/world1/layers/0/hitbox.png diff --git a/Assets/world/world1/layers/0/normal.png b/Assets/world/world1/layers/0/normal.png Binary files differnew file mode 100644 index 0000000..bc2ede6 --- /dev/null +++ b/Assets/world/world1/layers/0/normal.png diff --git a/Assets/world/world1/layers/0/texture.png b/Assets/world/world1/layers/0/texture.png Binary files differnew file mode 100644 index 0000000..429107e --- /dev/null +++ b/Assets/world/world1/layers/0/texture.png diff --git a/Assets/world/world1/layers/1/hitbox.png b/Assets/world/world1/layers/1/hitbox.png Binary files differnew file mode 100644 index 0000000..0a9a37e --- /dev/null +++ b/Assets/world/world1/layers/1/hitbox.png diff --git a/Assets/world/world1/layers/1/normal.png b/Assets/world/world1/layers/1/normal.png Binary files differnew file mode 100644 index 0000000..bc2ede6 --- /dev/null +++ b/Assets/world/world1/layers/1/normal.png diff --git a/Assets/world/world1/layers/1/texture.png b/Assets/world/world1/layers/1/texture.png Binary files differnew file mode 100644 index 0000000..149273f --- /dev/null +++ b/Assets/world/world1/layers/1/texture.png diff --git a/Assets/world/world1/layers/deco/forestTileBack.png b/Assets/world/world1/layers/deco/forestTileBack.png Binary files differnew file mode 100644 index 0000000..c5e9dac --- /dev/null +++ b/Assets/world/world1/layers/deco/forestTileBack.png diff --git a/Assets/world/world1/layers/deco/forestTileFront.png b/Assets/world/world1/layers/deco/forestTileFront.png Binary files differnew file mode 100644 index 0000000..234520c --- /dev/null +++ b/Assets/world/world1/layers/deco/forestTileFront.png diff --git a/Assets/world/world1/layers/deco/forestTileMid.png b/Assets/world/world1/layers/deco/forestTileMid.png Binary files differnew file mode 100644 index 0000000..b51d0bc --- /dev/null +++ b/Assets/world/world1/layers/deco/forestTileMid.png diff --git a/Assets/world/world1/layers/deco/normal.png b/Assets/world/world1/layers/deco/normal.png Binary files differnew file mode 100644 index 0000000..767f7e3 --- /dev/null +++ b/Assets/world/world1/layers/deco/normal.png diff --git a/Assets/world/world1/layers/deco/texture.png b/Assets/world/world1/layers/deco/texture.png Binary files differnew file mode 100644 index 0000000..ce37a03 --- /dev/null +++ b/Assets/world/world1/layers/deco/texture.png @@ -103,3 +103,4 @@ lib/libsoil.a: .PHONY: all remake clean cleaner cleanall resources +.PHONY: all remake clean cleaner resources mem diff --git a/Scripts/init.lua b/Scripts/init.lua index 66fbcb1..60f21c6 100644 --- a/Scripts/init.lua +++ b/Scripts/init.lua @@ -4,20 +4,24 @@ player = { Player = 0, EventListeners = { MoveLeftPressed = function(self) - self.Velocity.x = self.Velocity.x - 3.0 + --self.Velocity.x = self.Velocity.x - 3.0 + self.Velocity.y = self.Velocity.y - 1.0 self.Render.flipx = true; end, MoveLeftReleased = function(self) -- TODO can't put text at world coordinates right now --game.puts("default", self.Position.x, self.Position.y+100, "Hey. Hag?") - self.Velocity.x = self.Velocity.x + 3.0 + --self.Velocity.x = self.Velocity.x + 3.0 + self.Velocity.y = self.Velocity.y + 1.0 end, MoveRightPressed = function(self) - self.Velocity.x = self.Velocity.x + 3.0 + --self.Velocity.x = self.Velocity.x + 3.0 + self.Velocity.y = self.Velocity.y + 1.0 self.Render.flipx = false; end, MoveRightReleased = function(self) - self.Velocity.x = self.Velocity.x - 3.0 + --self.Velocity.x = self.Velocity.x - 3.0 + self.Velocity.y = self.Velocity.y - 1.0 end, JumpKeyPressed = function(self) if self.Physics.standing == true then @@ -29,14 +33,20 @@ player = { end }, Position = { - x = 15, - y = 75 + 15.0, 20.0 }, Velocity = { x = 0.0, y = 0.0 }, - Physics = 0, + Physics = { + hitbox = { + ll = {x = -0.5, y = -0.8}, + lr = {x = 0.5, y = -0.8}, + ul = {x = -0.5, y = 0.8}, + ur = {x = 0.5, y = 0.8}, + } + }, Name = "bord", Audio = { file = "Assets/jump.wav" @@ -45,7 +55,13 @@ player = { hellofalse = false, Render = { texture = "Assets/player.png", - visible = true + visible = true, + offset = { + ll = {x = -0.5, y = -0.8}, + lr = {x = 0.5, y = -0.8}, + ul = {x = -0.5, y = 0.8}, + ur = {x = 0.5, y = 0.8}, + } }, Light = { r = 1.0, @@ -54,11 +70,6 @@ player = { strength = 1.0 }, Idle = function(self) - --if (self.visibleTick >= 20) then - -- self.Render.visible = not self.Render.visible - -- self.visibleTick = 0 - --end - --self.visibleTick = self.visibleTick + 1 end, visibleTick = 0 } diff --git a/Scripts/world.lua b/Scripts/world.lua index bb6c61e..1fb74d2 100644 --- a/Scripts/world.lua +++ b/Scripts/world.lua @@ -1,112 +1,31 @@ -world = { - Seed = 5345345, - Layers = 2, - - -- This is run when the world is registered and not after, - -- although it is possible to register materials later +newWorld = { Register = function(self) - - -- TODO make world have global textures to speed up rendering - - self:registerMaterial("grass", { - -- TODO combine both of these into 1 - texture = { - file = "Assets/world.png", - offset = { x = 0, y = 0 }, - size = { x = 64, y = 64 } - }, - normal = { - file = "Assets/world_normal.png", - offset = { x = 0, y = 0 }, - size = { x = 64, y = 64 } - } + self.unitSize = 8; + self:createLayer(0, { + texture = { file = "Assets/world/world1/layers/0/texture.png" }, + normal = { file = "Assets/world/world1/layers/0/normal.png" }, + hitbox = "Assets/world/world1/layers/0/hitbox.png" }); - self:registerMaterial("dirt", { - texture = { - file = "Assets/world.png", - offset = { x = 64, y = 0 }, - size = { x = 64, y = 64 } - }, - normal = { - file = "Assets/world_normal.png", - offset = { x = 64, y = 0 }, - size = { x = 64, y = 64 } - } + self:createLayer(1, { + texture = { file = "Assets/world/world1/layers/1/texture.png" }, + normal = { file = "Assets/world/world1/layers/1/normal.png" }, + hitbox = "Assets/world/world1/layers/1/hitbox.png" }); - self:registerMaterial("stone", { - texture = { - file = "Assets/world.png", - offset = { x = 128, y = 0 }, - size = { x = 64, y = 64 } - }, - normal = { - file = "Assets/world_normal.png", - offset = { x = 128, y = 0 }, - size = { x = 64, y = 64 } - } + self:createDecoLayer(7, { + texture = { file = "Assets/world/world1/layers/deco/forestTileBack.png" }, + normal = { file = "Assets/world/world1/layers/deco/normal.png" }, }); - self:registerMaterial("flower", { - texture = { - file = "Assets/world.png", - offset = { x = 192, y = 0 }, - size = { x = 64, y = 64 } - }, - normal = { - file = "Assets/world_normal.png", - offset = { x = 192, y = 0 }, - size = { x = 64, y = 64 } - }, - passable = true + self:createDecoLayer(5.5, { + texture = { file = "Assets/world/world1/layers/deco/forestTileMid.png" }, + normal = { file = "Assets/world/world1/layers/deco/normal.png" }, }); - self:registerMaterial("trunk", { - texture = { - file = "Assets/world.png", - offset = { x = 256, y = 0 }, - size = { x = 64, y = 64 } - }, - normal = { - file = "Assets/world_normal.png", - offset = { x = 256, y = 0 }, - size = { x = 64, y = 64 } - } + self:createDecoLayer(4, { + texture = { file = "Assets/world/world1/layers/deco/forestTileFront.png" }, + normal = { file = "Assets/world/world1/layers/deco/normal.png" }, }); end, - Generate = function(self) - math.randomseed(self.Seed) - xsize, ysize, zsize = self:setSize(250, 128, 3) - for Z = 0,zsize-1 do - for X = 0,xsize-1 do - if Z == 0 then - YGen = math.floor(6*math.sin(X/20)) + 64 - elseif Z == 1 then - YGen = math.floor(9*math.sin(X/20)) + 64 - else - YGen = math.floor(15*math.sin(X/20)) + 64 - end - YDepth = math.random(3,5) - for Y = 0,ysize-1 do - if Y == YGen then - self:setData(X, Y, Z, "grass"); - elseif Y < YGen and Y > (YGen - YDepth) then - self:setData(X, Y, Z, "dirt"); - elseif Y < YGen then - self:setData(X, Y, Z, "stone"); - elseif Y == YGen + 1 then - if math.random(0, 100) == 53 then - self:setData(X, Y, Z, "flower"); - elseif math.random(0, 100) == 45 then - self:setData(X, Y, Z, "trunk"); - end - end - --print(X..","..Y..","..Z); - end - end - end - --self:setData(1000, 1345, 5, "grass"); -- Test error checking - print("Done with world gen"); end } ---world:Generate() -game.worldRegister(world) +game.worldRegister(newWorld); diff --git a/lib/sol2 b/lib/sol2 deleted file mode 160000 -Subproject 4de99c5b41b64b7e654bf8e48b177e8414a756b diff --git a/src/components/Component.hpp b/src/components/Component.hpp index 576a059..3075cea 100644 --- a/src/components/Component.hpp +++ b/src/components/Component.hpp @@ -24,6 +24,10 @@ #include <entityx/entityx.h> #include <sol/sol.hpp> +#include <glm/glm.hpp> + +#include <script/vectors.hpp> + template<typename T> class Component : public entityx::Component<T> { diff --git a/src/components/Physics.hpp b/src/components/Physics.hpp index edf5ac5..2d6b9b2 100644 --- a/src/components/Physics.hpp +++ b/src/components/Physics.hpp @@ -25,8 +25,38 @@ struct Physics : Component<Physics> { public: bool standing = true; + bool gravity = true; + glm::vec2 corners[4] = { + glm::vec2(-0.5, -0.5), // lower left + glm::vec2( 0.5, -0.5), // lower right + glm::vec2(-0.5, 0.5), // upper left + glm::vec2( 0.5, 0.5) // upper right + }; + Physics FromLua([[maybe_unused]] sol::object ref) { + if (ref.get_type() == sol::type::table) { + sol::table tab = ref; + + if (tab["gravity"].get_type() == sol::type::boolean) + this->gravity = tab["gravity"]; + + if (tab["hitbox"].get_type() == sol::type::table) { + sol::table hitbox = tab["hitbox"]; + if (hitbox["ll"] == sol::type::table) + corners[0] = Script::to<glm::vec2>(hitbox["ll"]); + if (hitbox["lr"] == sol::type::table) + corners[1] = Script::to<glm::vec2>(hitbox["lr"]); + if (hitbox["ul"] == sol::type::table) + corners[2] = Script::to<glm::vec2>(hitbox["ul"]); + if (hitbox["ur"] == sol::type::table) + corners[3] = Script::to<glm::vec2>(hitbox["ur"]); + } + } else { + throw std::string( + "Physics component table formatted incorrectly" + ); + } return *this; } diff --git a/src/components/Position.hpp b/src/components/Position.hpp index fcd62f8..a0adff5 100644 --- a/src/components/Position.hpp +++ b/src/components/Position.hpp @@ -24,25 +24,26 @@ struct Position : Component<Position> { public: - double x, y; + float x, y, z; - Position(double _x = 0, double _y = 0) : - x(_x), y(_y) {} + Position(float _x = 0.0f, float _y = 0.0f, float _z = 0.0f) : + x(_x), y(_y), z(_z) {} Position FromLua(sol::object ref) { - if (ref.get_type() == sol::type::table) { - sol::table tab = ref; - if (tab["x"] != nullptr) - this->x = tab["x"]; - if (tab["y"] != nullptr) - this->y = tab["y"]; - } else { - throw std::string("Position table not formatted properly"); - } + glm::vec3 vec = Script::to<glm::vec3>(ref); + this->x = vec.x; + this->y = vec.y; + this->z = vec.z; + return *this; } + glm::vec3 vec() + { + return glm::vec3(x, y, z); + } + void serialize(cereal::JSONOutputArchive& ar) final { ar(CEREAL_NVP(x), CEREAL_NVP(y)); } diff --git a/src/components/Render.hpp b/src/components/Render.hpp index 93be5d8..10a04fc 100644 --- a/src/components/Render.hpp +++ b/src/components/Render.hpp @@ -28,6 +28,12 @@ public: Texture normal; bool visible; bool flipX = false; + glm::vec2 corners[4] = { + glm::vec2(-0.5, -0.5), // lower left + glm::vec2( 0.5, -0.5), // lower right + glm::vec2(-0.5, 0.5), // upper left + glm::vec2( 0.5, 0.5) // upper right + }; Render(std::string _file) : texture(_file), visible(true) {} @@ -46,6 +52,19 @@ public: this->normal = Texture(tab.get<std::string>("normal")); if (tab["flipx"].get_type() == sol::type::boolean) this->flipX = tab["flipx"]; + + if (tab["offset"].get_type() == sol::type::table) { + sol::table offset = tab["offset"]; + if (offset["ll"] == sol::type::table) + corners[0] = Script::to<glm::vec2>(offset["ll"]); + if (offset["lr"] == sol::type::table) + corners[1] = Script::to<glm::vec2>(offset["lr"]); + if (offset["ul"] == sol::type::table) + corners[2] = Script::to<glm::vec2>(offset["ul"]); + if (offset["ur"] == sol::type::table) + corners[3] = Script::to<glm::vec2>(offset["ur"]); + } + } else { throw std::string( "Render component table formatted incorrectly" diff --git a/src/components/Velocity.hpp b/src/components/Velocity.hpp index f48a9f3..420dd3d 100644 --- a/src/components/Velocity.hpp +++ b/src/components/Velocity.hpp @@ -25,25 +25,24 @@ struct Velocity : Component<Velocity> { public: - double x, y; + float x, y; - Velocity(double _x = 0, double _y = 0) : + Velocity(float _x = 0, float _y = 0) : x(_x), y(_y) {} Velocity FromLua(sol::object ref) { - if (ref.get_type() == sol::type::table) { - sol::table tab = ref; - if (tab["x"] != nullptr) - this->x = tab["x"]; - if (tab["y"] != nullptr) - this->y = tab["y"]; - } else { - throw std::string("Velocity table not formatted properly"); - } + glm::vec2 vel = Script::to<glm::vec2>(ref); + this->x = vel.x; + this->y = vel.y; return *this; } + glm::vec2 vec() + { + return glm::vec2(x, y); + } + void serialize(cereal::JSONOutputArchive& ar) final { ar(CEREAL_NVP(x), CEREAL_NVP(y)); } diff --git a/src/engine.cpp b/src/engine.cpp index dc0c481..a891e1e 100644 --- a/src/engine.cpp +++ b/src/engine.cpp @@ -30,6 +30,7 @@ #include "render.hpp" #include "physics.hpp" #include "text.hpp" +#include "ui.hpp" #include "components/EventListener.hpp" #include "components/Script.hpp" @@ -150,6 +151,7 @@ void Engine::renderLoop(int& fpsCounter) entityx::TimeDelta dt = 0; /**< Elapsed milliseconds since each loop */ while (shouldRun()) { systems.update<TextSystem>(dt); + systems.update<UISystem>(dt); systems.update<RenderSystem>(dt); fpsCounter++; } @@ -159,21 +161,26 @@ void Engine::run(void) { int fpsCounter = 0; - // Start logic thread - logicThread = std::thread([this](void) { - logicLoop(); - }); - - physicsThread = std::thread([this](void) { - physicsLoop(); - }); + logicThread = std::thread([this] { logicLoop(); }); + physicsThread = std::thread([this] { physicsLoop(); }); - debugThread = std::thread([this, &fpsCounter](void) { + debugThread = std::thread([this, &fpsCounter] { while (shouldRun()) { - std::this_thread::sleep_for(1s); - fps = fpsCounter; + std::this_thread::sleep_for(250ms); + fps = fpsCounter*4; fpsCounter = 0; - systems.system<TextSystem>()->put("default", 0, 0, "fps: "s + std::to_string(fps)); + + systems.system<TextSystem>()-> + put("default", 0, 0, "fps: "s + std::to_string(fps)); + + entities.each<Player, Position>( + [this](entityx::Entity, Player &p, Position &pos){ + (void)p; + std::string pr = "pos: " + std::to_string(pos.x) + + "," + std::to_string(pos.y); + systems.system<TextSystem>()->put("default", 0, 24, pr); + + }); } }); diff --git a/src/events/render.hpp b/src/events/render.hpp index bcecac6..7bb45df 100644 --- a/src/events/render.hpp +++ b/src/events/render.hpp @@ -1,6 +1,25 @@ +/* + * Copyright (C) 2019 Belle-Isle, Andrew <drumsetmonkey@gmail.com> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see <http://www.gnu.org/licenses/>. + */ + #ifndef EVENTS_RENDER_HPP_ #define EVENTS_RENDER_HPP_ +#include <GL/glew.h> + struct NewRenderEvent { GLuint vbo; @@ -8,9 +27,24 @@ struct NewRenderEvent GLuint normal; unsigned int vertex; - NewRenderEvent(GLuint _vbo, GLuint _tex, GLuint _normal, unsigned int _vertex) : + NewRenderEvent(GLuint _vbo, GLuint _tex, + GLuint _normal, unsigned int _vertex) : vbo(_vbo), tex(_tex), normal(_normal), vertex(_vertex) {} }; +struct WorldMeshUpdateEvent +{ + GLuint worldVBO; + GLuint worldTexture; + GLuint worldNormal; + unsigned int numVertex; + + WorldMeshUpdateEvent(GLuint v, GLuint t, + GLuint n, unsigned int p) : + worldVBO(v), worldTexture(t), worldNormal(n), numVertex(p) {} + + WorldMeshUpdateEvent() {}; +}; + #endif // EVENTS_RENDER_HPP_ diff --git a/src/events/world.hpp b/src/events/world.hpp index e5969c0..e6f6d79 100644 --- a/src/events/world.hpp +++ b/src/events/world.hpp @@ -28,16 +28,4 @@ struct WorldChangeEvent newWorld(w) {} }; -struct WorldMeshUpdateEvent -{ - GLuint worldVBO; - unsigned int numVertex; - GLuint worldTexture; - GLuint worldNormal; - - WorldMeshUpdateEvent(GLuint v, unsigned int p, - GLuint t, GLuint n) : - worldVBO(v), numVertex(p), worldTexture(t), worldNormal(n) {} -}; - #endif//EVENTS_WORLD_HPP diff --git a/src/input.hpp b/src/input.hpp index 6180388..8f7aa8f 100644 --- a/src/input.hpp +++ b/src/input.hpp @@ -33,7 +33,7 @@ struct KeyUpEvent SDL_Keycode sym; Uint16 mod; - KeyUpEvent(const SDL_Keysym& keysym) : + explicit KeyUpEvent(const SDL_Keysym& keysym) : sym(keysym.sym), mod(keysym.mod) {} }; @@ -45,10 +45,28 @@ struct KeyDownEvent { SDL_Keycode sym; Uint16 mod; - KeyDownEvent(const SDL_Keysym& keysym) : + explicit KeyDownEvent(const SDL_Keysym& keysym) : sym(keysym.sym), mod(keysym.mod) {} }; +struct MouseUpEvent { + Uint8 button; + Sint32 x; + Sint32 y; + + explicit MouseUpEvent(const SDL_MouseButtonEvent& mbe) : + button(mbe.button), x(mbe.x), y(mbe.y) {} +}; + +struct MouseDownEvent { + Uint8 button; + Sint32 x; + Sint32 y; + + explicit MouseDownEvent(const SDL_MouseButtonEvent& mbe) : + button(mbe.button), x(mbe.x), y(mbe.y) {} +}; + /** * @class InputSystem * Listens for user input from SDL, and emits input events accordingly. diff --git a/src/physics.cpp b/src/physics.cpp index 1f70ecc..4567288 100644 --- a/src/physics.cpp +++ b/src/physics.cpp @@ -38,29 +38,45 @@ void PhysicsSystem::update([[maybe_unused]]entityx::EntityManager& entities, bool has_phys = e.has_component<Physics>(); - pos.x += (vel.x * dt/1000.0); - pos.y += (vel.y * dt/1000.0); - - // TODO make this intersect world instead of 0 y + // If the entity has physics if (has_phys) { + Physics *p = e.component<Physics>().get(); - double fallPosition = currentWorld->getHeight(pos.x, pos.y, 0.0); + glm::vec3 start = pos.vec(); + + glm::vec3 goal = pos.vec(); + goal.x += (vel.x * dt/1000.0); + goal.y += (vel.y * dt/1000.0); + + glm::vec3 end = currentWorld->collide(start, goal, *p); + (void)end; + + //std::cout << end.x << "," << end.y << std::endl; + + pos.x = goal.x; + pos.y = goal.y; + pos.z = goal.z; + + //float fallPosition = currentWorld->getHeight(pos.x, pos.y, 0.0); - Physics *p = e.component<Physics>().get(); // TODO only make this occur when the entity has a hitbox - if (pos.y == fallPosition) { - p->standing = true; - return; - } + //if (pos.y == fallPosition) { + // p->standing = true; + // return; + //} - if (pos.y < fallPosition) { - pos.y = fallPosition; - vel.y = 0; - p->standing = true; - } else { - p->standing = false; - vel.y -= 32.2 * (dt/1000.0f); - } + //if (pos.y < fallPosition) { + // pos.y = fallPosition; + // vel.y = 0; + // p->standing = true; + //} else { + // p->standing = false; + //if (p->gravity) + // vel.y -= 32.2 * (dt/1000.0f); + //} + } else { + pos.x += (vel.x * dt/1000.0); + pos.y += (vel.y * dt/1000.0); } }); } diff --git a/src/render.cpp b/src/render.cpp index 0f0b138..6e30f21 100644 --- a/src/render.cpp +++ b/src/render.cpp @@ -69,6 +69,20 @@ void RenderSystem::update([[maybe_unused]] entityx::EntityManager& entities, glEnable(GL_CULL_FACE); glEnable(GL_POLYGON_OFFSET_FILL); + /************ + * CAMERA * + ************/ + try { + if (player.has_component<Position>()) { + Position *pos = player.component<Position>().get(); + camPos.y = pos->y; + camPos.x = pos->x; + } + } catch (...) { // If the player doesn't exist or anything goes wrong + camPos.y = 0.0f; + camPos.x = 0.0f; + } + glm::mat4 view = glm::lookAt(camPos, // Pos camPos + rot, // Facing @@ -101,7 +115,6 @@ void RenderSystem::update([[maybe_unused]] entityx::EntityManager& entities, glUseProgram(s); - glUniformMatrix4fv(v, 1, GL_FALSE, glm::value_ptr(view)); glUniformMatrix4fv(p, 1, GL_FALSE, glm::value_ptr(projection)); glUniformMatrix4fv(m, 1, GL_FALSE, glm::value_ptr(model)); @@ -114,20 +127,6 @@ void RenderSystem::update([[maybe_unused]] entityx::EntityManager& entities, GLfloat amb[4] = {1.0f, 1.0f, 1.0f, 0.0f}; glUniform4fv(b, 1, amb); - /************ - * CAMERA * - ************/ - try { - if (player.has_component<Position>()) { - Position *pos = player.component<Position>().get(); - camPos.y = pos->y; - camPos.x = pos->x; - } - } catch (...) { // If the player doesn't exist or anything goes wrong - camPos.y = 0.0f; - camPos.x = 0.0f; - } - /************** * LIGHTING * **************/ @@ -167,19 +166,18 @@ void RenderSystem::update([[maybe_unused]] entityx::EntityManager& entities, //if (e.has_component<Scripted>()) { // e.component<Scripted>()->updateRender(); //} - - float w = 0.5f; - float h = (float)rend.texture.height/rend.texture.width; + + auto& c = rend.corners; GLuint tri_vbo; GLfloat tri_data[] = { - (float)p.x-w, (float)p.y , 0.0f, 0.0f, 1.0f, 1.0f, - (float)p.x+w, (float)p.y , 0.0f, 1.0f, 1.0f, 1.0f, - (float)p.x-w, (float)p.y+h, 0.0f, 0.0f, 0.0f, 1.0f, - - (float)p.x+w, (float)p.y , 0.0f, 1.0f, 1.0f, 1.0f, - (float)p.x+w, (float)p.y+h, 0.0f, 1.0f, 0.0f, 1.0f, - (float)p.x-w, (float)p.y+h, 0.0f, 0.0f, 0.0f, 1.0f, + p.x+c[0].x, p.y+c[0].y, 0.0f, 0.0f, 1.0f, 1.0f, + p.x+c[1].x, p.y+c[1].y, 0.0f, 1.0f, 1.0f, 1.0f, + p.x+c[2].x, p.y+c[2].y, 0.0f, 0.0f, 0.0f, 1.0f, + + p.x+c[1].x, p.y+c[1].y, 0.0f, 1.0f, 1.0f, 1.0f, + p.x+c[3].x, p.y+c[3].y, 0.0f, 1.0f, 0.0f, 1.0f, + p.x+c[2].x, p.y+c[2].y, 0.0f, 0.0f, 0.0f, 1.0f, }; bool flipped = false; @@ -219,24 +217,25 @@ void RenderSystem::update([[maybe_unused]] entityx::EntityManager& entities, }); glUniform1i(f, 0); - // If we were given a world VBO render it - if (worldVBO) { + for (auto& w : worldRenders) { + auto& layer = w.second; + glActiveTexture(GL_TEXTURE0); - glBindTexture(GL_TEXTURE_2D, worldTexture); + glBindTexture(GL_TEXTURE_2D, layer.tex); glUniform1i(q, 0); glActiveTexture(GL_TEXTURE1); - glBindTexture(GL_TEXTURE_2D, worldNormal); + glBindTexture(GL_TEXTURE_2D, layer.normal); glUniform1i(n, 1); - glBindBuffer(GL_ARRAY_BUFFER, worldVBO); + glBindBuffer(GL_ARRAY_BUFFER, w.first); glVertexAttribPointer(a, 3, GL_FLOAT, GL_FALSE, 6*sizeof(float), 0); glVertexAttribPointer(t, 2, GL_FLOAT, GL_FALSE, 6*sizeof(float), (void*)(3*sizeof(float))); glVertexAttribPointer(r, 1, GL_FLOAT, GL_FALSE, 6*sizeof(float), (void*)(5*sizeof(float))); - glDrawArrays(GL_TRIANGLES, 0, worldVertex); + glDrawArrays(GL_TRIANGLES, 0, layer.vertex); } glDisableVertexAttribArray(a); @@ -346,7 +345,7 @@ int RenderSystem::init(void) return -1; } - SDL_GL_SetSwapInterval(0); + SDL_GL_SetSwapInterval(1); glEnable(GL_BLEND); glEnable(GL_DEPTH_TEST); @@ -407,10 +406,12 @@ void RenderSystem::receive(const NewRenderEvent &nre) void RenderSystem::receive(const WorldMeshUpdateEvent &wmu) { - worldVBO = wmu.worldVBO; - worldVertex = wmu.numVertex; - worldTexture = wmu.worldTexture; - worldNormal = wmu.worldNormal; + worldRenders.insert_or_assign( + wmu.worldVBO, + WorldRenderData(wmu.worldTexture, + wmu.worldNormal, + wmu.numVertex) + ); } void RenderSystem::receive(const entityx::ComponentAddedEvent<Player> &cae) diff --git a/src/render.hpp b/src/render.hpp index c4456cf..45fb548 100644 --- a/src/render.hpp +++ b/src/render.hpp @@ -37,8 +37,8 @@ #include "shader.hpp" #include "world.hpp" #include "components/Player.hpp" + #include "events/render.hpp" -#include "events/world.hpp" #include <map> @@ -52,6 +52,16 @@ struct UIRenderData tex(_tex), normal(_normal), vertex(_vertex) {} }; +struct WorldRenderData +{ + GLuint tex; + GLuint normal; + unsigned int vertex; + + WorldRenderData(GLuint _tex, GLuint _normal, unsigned int _vertex) : + tex(_tex), normal(_normal), vertex(_vertex) {} +}; + class RenderSystem : public entityx::System<RenderSystem>, public entityx::Receiver<RenderSystem> { @@ -69,11 +79,8 @@ private: // Map of VBOs and their render data std::map<GLuint, UIRenderData> uiRenders; + std::map<GLuint, WorldRenderData> worldRenders; - GLuint worldVBO = 0; - unsigned int worldVertex = 0; - GLuint worldTexture = 0; - GLuint worldNormal = 0; entityx::Entity player; // Save the player so we can track the camera public: RenderSystem() : diff --git a/src/script.cpp b/src/script.cpp index 9fae1c9..b1d82e9 100644 --- a/src/script.cpp +++ b/src/script.cpp @@ -89,9 +89,10 @@ void ScriptSystem::doFile(void) void ScriptSystem::scriptExport(void) { lua.new_usertype<Position>("Position", - sol::constructors<Position(double x, double y), Position()>(), + sol::constructors<Position(float x, float y, float z), Position()>(), "x", &Position::x, - "y", &Position::y); + "y", &Position::y, + "z", &Position::z); lua.new_usertype<Name>("Name", sol::constructors<Name(std::string), Name()>(), @@ -104,7 +105,7 @@ void ScriptSystem::scriptExport(void) "flipx", &Render::flipX); lua.new_usertype<Velocity>("Velocity", - sol::constructors<Velocity(double, double), Velocity()>(), + sol::constructors<Velocity(float, float), Velocity()>(), "x", &Velocity::x, "y", &Velocity::y); @@ -120,7 +121,8 @@ void ScriptSystem::scriptExport(void) lua.new_usertype<Physics>("Physics", sol::constructors<Physics(void), Physics()>(), - "standing", &Physics::standing); + "standing", &Physics::standing, + "gravity", &Physics::gravity); lua.new_usertype<Audio>("Audio", sol::constructors<Audio(std::string)>(), @@ -130,10 +132,12 @@ void ScriptSystem::scriptExport(void) sol::constructors<World(sol::object), World(void)>(), "Generate", &World::generate, "Seed", sol::property(&World::setSeed, &World::getSeed), - "setData", &World::setData, - "registerMaterial", &World::registerMaterial, - "setSize", &World::setSize, - "getSize", &World::getSize); + "getSize", &World::getSize, + + // New stuff + "unitSize", sol::property(&World::setUnitSize, &World::getUnitSize), + "createLayer", &World::registerLayer, + "createDecoLayer", &World::registerDecoLayer); game = lua["game"].get_or_create<sol::table>(); game.set_function("spawn", bindInstance(&ScriptSystem::spawn, this)); @@ -143,7 +147,7 @@ void ScriptSystem::scriptExport(void) sol::table ScriptSystem::spawn(sol::object param) { - sol::table* toRet; // "Entitiy" table to be returned + sol::table* toRet = NULL; // "Entitiy" table to be returned if (param.get_type() == sol::type::table) { sol::table tab = param; // Cast the generic parameter to a table diff --git a/src/script/vectors.cpp b/src/script/vectors.cpp new file mode 100644 index 0000000..aa27aad --- /dev/null +++ b/src/script/vectors.cpp @@ -0,0 +1,108 @@ +/* + * Copyright (C) 2019 Belle-Isle, Andrew <drumsetmonkey@gmail.com> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see <http://www.gnu.org/licenses/>. + */ + +#include "vectors.hpp" + +#include <glm/glm.hpp> + +#include <iostream> + +namespace Script +{ + template<class T> + T to(sol::object obj) + { + (void)obj; + T fake; + return fake; + } + + template<> + glm::vec2 to<glm::vec2>(sol::object obj) + { + glm::vec2 toReturn(0.0f); + + if (obj.get_type() == sol::type::table) { + sol::table table = obj; + // X + if (table["x"] == sol::type::number) { + toReturn.x = table["x"]; + } else if (table[1] == sol::type::number) { + toReturn.x = table[1]; + } + // Y + if (table["y"] == sol::type::number) { + toReturn.y = table["y"]; + } else if (table[2] == sol::type::number) { + toReturn.y = table[2]; + } + } else { + std::cerr << "Vectors must be in table form" << std::endl; + } + + return toReturn; + } + + template<> + glm::vec3 to<glm::vec3>(sol::object obj) + { + glm::vec3 toReturn(0.0f); + + if (obj.get_type() == sol::type::table) { + sol::table table = obj; + glm::vec2 base = to<glm::vec2>(table); + toReturn.x = base.x; + toReturn.y = base.y; + + // Z + if (table["z"] == sol::type::number) { + toReturn.z = table["z"]; + } else if (table[3] == sol::type::number) { + toReturn.z = table[3]; + } + } else { + std::cerr << "Vectors must be in table form" << std::endl; + } + + return toReturn; + } + + template<> + glm::vec4 to<glm::vec4>(sol::object obj) + { + glm::vec4 toReturn(0.0f); + + if (obj.get_type() == sol::type::table) { + sol::table table = obj; + glm::vec3 base = to<glm::vec3>(table); + toReturn.x = base.x; + toReturn.y = base.y; + toReturn.z = base.z; + + // W + if (table["w"] == sol::type::number) { + toReturn.w = table["w"]; + } else if (table[4] == sol::type::number) { + toReturn.w = table[4]; + } + } else { + std::cerr << "Vectors must be in table form" << std::endl; + } + + return toReturn; + } +} diff --git a/src/script/vectors.hpp b/src/script/vectors.hpp new file mode 100644 index 0000000..9ee0b31 --- /dev/null +++ b/src/script/vectors.hpp @@ -0,0 +1,31 @@ +/** + * @file vectors.hpp + * + * Copyright (C) 2019 Belle-Isle, Andrew <drumsetmonkey@gmail.com> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see <http://www.gnu.org/licenses/>. + */ + +#ifndef SCRIPT_VECTORS_HPP_ +#define SCRIPT_VECTORS_HPP_ + +#include <sol/sol.hpp> + +namespace Script +{ + template<class T> + T to(sol::object obj); +} + +#endif//SCRIPT_VECTORS_HPP_ diff --git a/src/text.cpp b/src/text.cpp index fb82875..5490224 100644 --- a/src/text.cpp +++ b/src/text.cpp @@ -13,13 +13,15 @@ TextSystem::~TextSystem(void) } void TextSystem::configure([[maybe_unused]] entityx::EntityManager& entities, - [[maybe_unused]] entityx::EventManager& events) + entityx::EventManager& events) { + shouldUpdateVBOs = false; + + events.subscribe<ShowTextEvent>(*this); + if (FT_Init_FreeType(&freetype) != 0) { // TODO handle error } - - shouldUpdateVBOs = false; } /** @@ -42,6 +44,11 @@ void TextSystem::update([[maybe_unused]] entityx::EntityManager& entites, } } +void TextSystem::receive(const ShowTextEvent& ste) +{ + put(ste.font, ste.x, ste.y, ste.text); +} + void TextSystem::loadFont(const std::string& name, const std::string& file, int size) @@ -80,8 +87,8 @@ void TextSystem::loadFont(const std::string& name, glGenTextures(1, &font.tex); glBindTexture(GL_TEXTURE_2D, font.tex); - glTexImage2D(GL_TEXTURE_2D, 0, GL_RED, width, height, - 0, GL_RED, GL_UNSIGNED_BYTE, 0); + glTexImage2D(GL_TEXTURE_2D, 0, GL_INTENSITY8, width, height, + 0, GL_LUMINANCE, GL_UNSIGNED_BYTE, 0); glPixelStorei(GL_UNPACK_ALIGNMENT, 1); @@ -113,7 +120,7 @@ void TextSystem::loadFont(const std::string& name, auto* g = face->glyph; glTexSubImage2D(GL_TEXTURE_2D, 0, offsetX, offsetY, g->bitmap.width, g->bitmap.rows, - GL_RED, GL_UNSIGNED_BYTE, + GL_LUMINANCE, GL_UNSIGNED_BYTE, g->bitmap.buffer); auto& d = font.data[c-32]; @@ -137,16 +144,21 @@ void TextSystem::put(const std::string& font, { if (fontData.find(font) == fontData.end()) return; - - y -= fontData[font].fontSize; - auto& vector = fontData[font].text; - if (auto i = std::find_if(vector.begin(), vector.end(), [&x, &y](const Text& t) { - return t.x == x && t.y == y; }); i != vector.end()) { - vector.erase(i); + + y = -(y + fontData[font].fontSize); + + const auto it = std::find_if(vector.begin(), vector.end(), + [&x, &y](const Text& t) { + return t.x == static_cast<int>(x) && t.y == static_cast<int>(y); + }); + if (it != vector.end()) { + *it = Text(text, x, y); + } else { + // Invert y axis so positive grows south. + fontData[font].text.emplace_back(text, x, y); } - fontData[font].text.emplace_back(text, x, y, -9.0f); shouldUpdateVBOs = true; } diff --git a/src/text.hpp b/src/text.hpp index 1ef2afa..0f98688 100644 --- a/src/text.hpp +++ b/src/text.hpp @@ -50,8 +50,8 @@ struct FT_Info { struct Text { std::string text; - float x; - float y; + int x; + int y; float z; Text(std::string _text, float _x, float _y, float _z = 0.0f) : @@ -74,7 +74,20 @@ struct Font { std::basic_string<TextMeshData> buffer; }; -class TextSystem : public entityx::System<TextSystem> +struct ShowTextEvent +{ + std::string font; + float x; + float y; + std::string text; + + explicit ShowTextEvent(const std::string& _font, float _x, float _y, + const std::string& _text) : + font(_font), x(_x), y(_y), text(_text) {} +}; + +class TextSystem : public entityx::System<TextSystem>, + public entityx::Receiver<TextSystem> { public: ~TextSystem(void); @@ -92,6 +105,8 @@ public: entityx::EventManager& events, entityx::TimeDelta dt) final; + void receive(const ShowTextEvent&); + void put(const std::string& font, float x, float y, const std::string& text); void loadFont(const std::string& name, const std::string& file, int size); diff --git a/src/texture.cpp b/src/texture.cpp index 2870b9f..331413b 100644 --- a/src/texture.cpp +++ b/src/texture.cpp @@ -21,9 +21,13 @@ #include "texture.hpp" +#include <soil/SOIL.h> + #include <unordered_map> #include <iostream> +#include <script/vectors.hpp> + struct TextureData { GLuint tex = 0; @@ -99,20 +103,10 @@ Texture::Texture(sol::object param) else return; // If we don't have image data just return a null image - if (tab["offset"] == sol::type::table) { - sol::table off = tab["offset"]; - if (off["x"] == sol::type::number) - offset.x = off.get<float>("x")/width; - if (off["y"] == sol::type::number) - offset.y = off.get<float>("y")/height; - } - - if (tab["size"] == sol::type::table) { - sol::table siz = tab["size"]; - if (siz["x"] == sol::type::number) - size.x = siz.get<float>("x")/width; - if (siz["y"] == sol::type::number) - size.y = siz.get<float>("y")/height; - } + if (tab["offset"].get_type() == sol::type::table) + offset = Script::to<glm::vec2>(tab["offset"]); + + if (tab["size"].get_type() == sol::type::table) + size = Script::to<glm::vec2>(tab["size"]); } } diff --git a/src/texture.hpp b/src/texture.hpp index 3daebbd..b03d649 100644 --- a/src/texture.hpp +++ b/src/texture.hpp @@ -20,8 +20,6 @@ #ifndef TEXTURE_HPP_ #define TEXTURE_HPP_ -#include <soil/SOIL.h> - #include <sol/sol.hpp> #include <GL/glew.h> diff --git a/src/ui.cpp b/src/ui.cpp new file mode 100644 index 0000000..71cb875 --- /dev/null +++ b/src/ui.cpp @@ -0,0 +1,56 @@ +#include "ui.hpp" + +#include "events/render.hpp" +#include "text.hpp" + +#include <SDL2/SDL_opengl.h> +#include <iostream> +#include <string> + +static NewRenderEvent NRE (0, 0, 0, 0); +static const unsigned int NRE_TEX_DATA = 0xBBBBBBBB; +static std::basic_string<TextMeshData> buffer; + +void UISystem::configure(entityx::EntityManager&, entityx::EventManager&) +{ +} + +void UISystem::update(entityx::EntityManager&, + entityx::EventManager& events, + entityx::TimeDelta) +{ + static bool t = false; + + if (!t) { + t = true; + + glGenBuffers(1, &NRE.vbo); + glGenTextures(1, &NRE.tex); + + glBindTexture(GL_TEXTURE_2D, NRE.tex); + glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA8, 1, 1, 0, GL_RGBA, GL_UNSIGNED_BYTE, &NRE_TEX_DATA); + + glPixelStorei(GL_UNPACK_ALIGNMENT, 1); + glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT); + glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT); + glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); + glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); + + buffer += { 30, -50, -5, 0, 0, 1 }; + buffer += { 160, -50, -5, 0, 0, 1 }; + buffer += { 160, -10, -5, 0, 0, 1 }; + buffer += { 160, -10, -5, 0, 0, 1 }; + buffer += { 30, -10, -5, 0, 0, 1 }; + buffer += { 30, -50, -5, 0, 0, 1 }; + + glBindBuffer(GL_ARRAY_BUFFER, NRE.vbo); + glBufferData(GL_ARRAY_BUFFER, + buffer.size() * sizeof(TextMeshData), buffer.data(), + GL_DYNAMIC_DRAW); + + NRE.normal = 0; + NRE.vertex = buffer.size(); + events.emit<NewRenderEvent>(NRE); + } +} + diff --git a/src/ui.hpp b/src/ui.hpp new file mode 100644 index 0000000..809b5dc --- /dev/null +++ b/src/ui.hpp @@ -0,0 +1,42 @@ +/** + * @file ui.hpp + * + * + * Copyright (C) 2022 Clyne Sullivan + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see <http://www.gnu.org/licenses/>. + */ + +#ifndef SYSTEM_UI_HPP_ +#define SYSTEM_UI_HPP_ + +#include <entityx/entityx.h> + +#include <map> +#include <string> +#include <tuple> +#include <vector> + +class UISystem : public entityx::System<UISystem> +{ +public: + void configure(entityx::EntityManager&, entityx::EventManager&) final; + + void update(entityx::EntityManager&, + entityx::EventManager&, + entityx::TimeDelta) final; +}; + +#endif // SYSTEM_UI_HPP_ + diff --git a/src/world.cpp b/src/world.cpp index cd89a22..69590e1 100644 --- a/src/world.cpp +++ b/src/world.cpp @@ -18,7 +18,7 @@ */ #include "world.hpp" -#include "events/render.hpp" + #include "events/world.hpp" /***************** @@ -55,131 +55,50 @@ World::World(sol::object param) // If a generate function is defined, call it if (generate != sol::nil) generate(this); - - // Create our world VBO - glGenBuffers(1, &worldVBO); - // Generate our mesh - generateMesh(); -} - -/* REGISTRY */ -void World::registerMaterial(std::string name, sol::object data) -{ - if (data.get_type() == sol::type::table) { - sol::table tab = data; - - // Make sure this material has not been registered before - auto it = string_registry.find(name); - if (it == string_registry.end()) { - string_registry.emplace(name, registry.size()); - registry.push_back(WorldMaterial(tab)); - } else { - std::cerr << "Material: " << name - << " was already registered" << std::endl; - } - } else { - // TODO better logging - std::cerr << "Material registration must have a table" << std::endl; - } -} - -/* DATA */ -void World::setData(unsigned int x, - unsigned int y, - unsigned int z, - std::string d) -{ - unsigned int discovered = -1; - - auto found = string_registry.find(d); - if (found != string_registry.end()) - discovered = found->second; - - try { - data.at(z).at(x).at(y) = discovered; - } catch (std::out_of_range &oor) { - // Make sure any assignments that are outsize specified world size are - // caught to avoid any seg faults - std::cerr << "Unable to set data at: " - << x << "," << y << "," << z - << " Exception: " << oor.what() << std::endl; - } -} - -/* SIZE */ -std::tuple<unsigned int, unsigned int, unsigned int> -World::setSize(unsigned int x, unsigned int y, unsigned int z) -{ - width = x; - height = y; - layers = z; - - data = std::vector<std::vector<std::vector<int>>> - (z, std::vector<std::vector<int>> - (x,std::vector<int> - (y, -1) - ) - ); - - return {width, height, layers}; } +// TODO std::tuple<unsigned int, unsigned int, unsigned int> World::getSize() { - return {width, height, layers}; + //return {width, height, layers}; + return {0, 0, 0}; } /* RENDERING */ void World::generateMesh() { - //const unsigned int voxelLength = 6; // 2 triangles @ 3 points each - - // Preallocate size of vertexes - mesh = std::basic_string<WorldMeshData>(); - for (float Z = data.size() - 1; Z >= 0; Z--) { - for (float X = 0; X < data.at(Z).size(); X++) { - for (float Y = 0; Y < data.at(Z).at(X).size(); Y++) { - int d = data.at(Z).at(X).at(Y); - - if (d == -1) // Don't make a mesh for air of course - continue; - - Texture &t = registry.at(d).texture; - glm::vec2& to = t.offset; - glm::vec2& ts = t.size; - - float tr = 1.0f; - - // TODO play with this a bit so it only goes trans - // if player is behind the front layer - try { - if (Z < data.size() - 1 && Z >= 0) { - if (data.at(Z+1).at(X).at(Y) == -1) - tr = 1.0f; - } - } catch (...) { - tr = 1.0f; - } - - mesh += {X , Y , Z, to.x , to.y+ts.y, tr}; - mesh += {X+1, Y , Z, to.x+ts.x, to.y+ts.y, tr}; - mesh += {X , Y+1, Z, to.x , to.y , tr}; - - mesh += {X+1, Y , Z, to.x+ts.x, to.y+ts.y, tr}; - mesh += {X+1, Y+1, Z, to.x+ts.x, to.y , tr}; - mesh += {X , Y+1, Z, to.x , to.y , tr}; - } - } + for (auto &l : drawLayers) { + + // Preallocate size of vertexes + + float Z = l->drawLayer; + auto to = l->texture.offset; + auto ts = l->texture.size; + float tr = 1.0f; + + float w = l->texture.width/unitSize; + float h = l->texture.height/unitSize; + + GLfloat mesh[36] = {0 , 0 , Z, to.x , to.y+ts.y, tr, + 0+w, 0 , Z, to.x+ts.x, to.y+ts.y, tr, + 0 , 0+h, Z, to.x , to.y , tr, + + 0+w, 0 , Z, to.x+ts.x, to.y+ts.y, tr, + 0+w, 0+h, Z, to.x+ts.x, to.y , tr, + 0 , 0+h, Z, to.x , to.y , tr}; + + glBindBuffer(GL_ARRAY_BUFFER, l->layerVBO); + glBufferData(GL_ARRAY_BUFFER, + 36 * sizeof(GLfloat), + mesh, + GL_STATIC_DRAW); + + meshAdd.push_back(WorldMeshUpdateEvent(l->layerVBO, + l->texture.tex, + l->normal.tex, + 36)); } - - glBindBuffer(GL_ARRAY_BUFFER, worldVBO); - glBufferData(GL_ARRAY_BUFFER, - mesh.size() * sizeof(WorldMeshData), - mesh.data(), - GL_STATIC_DRAW); - - meshUpdated = true; } /* SEED */ @@ -197,28 +116,156 @@ unsigned int World::setSeed(unsigned int s) /* PHYSICS */ double World::getHeight(double x, double y, double z) { - unsigned int X = static_cast<unsigned int>(x); - unsigned int Z = static_cast<unsigned int>(z); - - double Y = 0.0; - try { - auto &d = data.at(Z).at(X); - for (int yi = d.size()-1; yi >= 0; yi--) { - if (d.at(yi) >= 0) { - if (!registry.at(d.at(yi)).passable) { - Y = static_cast<double>(yi); - Y += 1; - break; + (void)y; + double Y = 0.0f; + for (auto &l : solidLayers) { + if (z == l->drawLayer) { + int wx = x*unitSize; + + int h = 0.0; + for (auto b : l->hitbox[wx]) { + if (b == true) + Y = h; + h++; + } + return ((Y+1)/unitSize); + } + } + return 0; +} + +bool World::isSolid(glm::vec3 pos) +{ + for (auto &l : solidLayers) { + if (pos.z == l->drawLayer) { + int wx = pos.x * unitSize; + int wy = pos.y * unitSize; + if (wx < 0 || wy < 0) return true; + + return l->hitbox[wx][wy]; + } + } + return false; +} + +std::vector<glm::vec3> +World::getIntersectingPlanes(glm::vec3 origin, Physics &phys) +{ + std::vector<glm::vec3> planes; + + glm::vec3 goal = origin; + + origin.x += phys.corners[0].x; + origin.y += phys.corners[0].y; + + goal.x += phys.corners[3].x; + goal.y += phys.corners[3].y; + + float step = 1.0f/unitSize; + for (;origin.y <= goal.y; origin.y += step){ + for (;origin.x <= goal.x; origin.x += step) { + if (isSolid(origin)) { + planes.push_back(origin); + } + } + } + + return planes; +} + +glm::vec3 World::collide(glm::vec3 &start, glm::vec3 &end, Physics &phys) +{ + // How far to push the entity to unintersect with the world + glm::vec3 push(0); + for (auto &l : solidLayers) { + if (start.z == l->drawLayer) { + glm::vec3 len = end-start; + glm::vec3 dir = glm::normalize(len); + float step = 1.0f/unitSize; + + glm::vec3 pos = start; + + for (float i = 0.0f; i < glm::length(len); i+=step, pos+=dir) { + // Get all colliding world spaces + std::vector<glm::vec3> inter = getIntersectingPlanes(pos, phys); + + //if (i == 0.0f) + // std::cout << inter.size() << std::endl; + + // If there are no colliding world spaces, don't bother + if (inter.size()) { + if (dir.x > 0.0f) { + // Moving to the right + int closest = inter.at(0).x; + for (auto &p : inter) { + if (p.x < closest) + closest = p.x; + } + push.x -= abs(closest - (pos.x + phys.corners[1].x)); + + } else if (dir.x < 0.0f) { + // Moving to the left + int closest = inter.at(0).x; + for (auto &p : inter) { + if (p.x > closest) + closest = p.x; + } + push.x += abs(closest - (pos.x + phys.corners[0].x)); + } + + if (dir.y > 0.0f) { + // Moving upwards + int closest = inter.at(0).y; + for (auto &p : inter) { + if (p.y < closest) + closest = p.y; + } + push.y -= abs(closest - (pos.y + phys.corners[2].y)); + } else if (dir.y < 0.0f) { + // Moving downwards + int closest = inter.at(0).y; + for (auto &p : inter) { + if (p.y > closest) + closest = p.y; + } + push.y += abs(closest - (pos.y + phys.corners[0].y)); + } + + if (push != glm::vec3(0.0f)) + return pos + push; } } } - } catch (...) { // If we get any errors, just let the character - //return y; - (void)y; - return 0.0; } + return end; +} + + +/********* +* NEW * +*********/ +void World::registerLayer(float z, sol::object obj) +{ + if (obj.get_type() == sol::type::table) { + sol::table tab = obj; + SolidLayer s(z, tab); + solidLayers.push_back(std::make_shared<SolidLayer>(s)); + drawLayers.push_back(std::make_shared<Layer>(s)); + } else { + throw std::string("Layer must receive a table"); + } + generateMesh(); +} - return Y; +void World::registerDecoLayer(float z, sol::object obj) +{ + if (obj.get_type() == sol::type::table) { + sol::table tab = obj; + drawLayers.push_back(std::make_shared<Layer>(Layer(z, tab))); + } else { + throw std::string("Layer must receive a table"); + } + generateMesh(); } @@ -247,12 +294,8 @@ void WorldSystem::update([[maybe_unused]]entityx::EntityManager& entities, events.emit<WorldChangeEvent>(currentWorld); } - if (currentWorld->meshUpdated) { - events.emit<WorldMeshUpdateEvent>( - currentWorld->worldVBO, - currentWorld->mesh.size(), - currentWorld->getTexture(), - currentWorld->getNormal() - ); + for (auto &ma : currentWorld->meshAdd) { + events.emit<WorldMeshUpdateEvent>(ma); } + currentWorld->meshAdd.clear(); } diff --git a/src/world.hpp b/src/world.hpp index 9314fa6..2d11706 100644 --- a/src/world.hpp +++ b/src/world.hpp @@ -26,14 +26,15 @@ #include <entityx/entityx.h> #include <sol/sol.hpp> +#include <soil/SOIL.h> + #include "texture.hpp" +#include "events/render.hpp" + +#include <components/Position.hpp> // For entity position +#include <components/Velocity.hpp> // For entity velocity +#include <components/Physics.hpp> // For entity hitbox(es) -struct WorldMeshData -{ - float posX, posY, posZ; - float texX, texY; - float transparency; -}__attribute__((packed)); struct WorldMaterial { @@ -57,6 +58,65 @@ struct WorldMaterial } }; +class Layer +{ + friend class World; +private: + Texture texture; + Texture normal; + + float drawLayer = 0.0f; + + GLuint layerVBO; +public: + + Layer(float z, sol::table tab) { + drawLayer = z; + if (tab["texture"] != nullptr) { + sol::object t = tab["texture"]; + texture = Texture(t); + } + if (tab["normal"] != nullptr) { + sol::object n = tab["normal"]; + normal = Texture(n); + } + + glGenBuffers(1, &layerVBO); + } +}; + +class SolidLayer : public Layer +{ + friend class World; +private: + std::vector<std::vector<bool>> hitbox; +public: + SolidLayer(float z, sol::table tab) : Layer(z, tab) { + if (tab["hitbox"] != nullptr) { + int width, height, channels; + unsigned char* box = + SOIL_load_image(std::string(tab["hitbox"]).c_str(), + &width, &height, &channels, + SOIL_LOAD_RGBA); + + for (int w = 0; w < width*channels; w+=channels) { + hitbox.push_back(std::vector<bool>(height)); + for (int h = 0; h < height; h++) { + unsigned char* c = &box[(w) + (width*h*channels)]; + // we want to read the last channel (alpha) + if (c[channels-1]) { + hitbox[w/channels][height-h] = true; + } + else + hitbox[w/channels][height-h] = false; + } + } + + SOIL_free_image_data(box); + } + } +}; + class World { friend class WorldSystem; @@ -64,19 +124,19 @@ private: unsigned int seed; unsigned int layers; - unsigned int height; - unsigned int width; + unsigned int unitSize; + std::vector<std::shared_ptr<SolidLayer>> solidLayers; + std::vector<std::shared_ptr<Layer>> drawLayers; - std::vector<std::vector<std::vector<int>>> data; + std::vector<glm::vec3> + getIntersectingPlanes(glm::vec3 origin, Physics &phys); - std::unordered_map<std::string, int> string_registry; - std::vector<WorldMaterial> registry; + bool isSolid(glm::vec3 pos); protected: // RENDER - std::basic_string<WorldMeshData> mesh; + std::vector<WorldMeshUpdateEvent> meshAdd; GLuint worldVBO; - bool meshUpdated = false; public: /* VARS */ sol::function generate; @@ -87,27 +147,12 @@ public: ~World() { registerMat = sol::nil; generate = sol::nil; - registry.clear(); - data.clear(); } - /* REGISTRY */ - void registerMaterial(std::string, sol::object); - - /* DATA */ - void setData(unsigned int, unsigned int, unsigned int, std::string); - - /* SIZE */ - std::tuple<unsigned int, unsigned int, unsigned int> setSize(unsigned int, - unsigned int, - unsigned int); std::tuple<unsigned int, unsigned int, unsigned int> getSize(); /* RENDERING */ void generateMesh(); - std::basic_string<WorldMeshData>& getMesh() {return mesh;} - GLuint getTexture() {return registry.at(0).texture.tex;} - GLuint getNormal() {return registry.at(0).normal.tex;}; /* SEED */ unsigned int getSeed(); @@ -115,6 +160,14 @@ public: /* PHYSICS */ double getHeight(double x, double y, double z); + glm::vec3 collide(glm::vec3 &start, glm::vec3 &end, Physics &phys); + + // NEW + unsigned int getUnitSize() {return unitSize;} + void setUnitSize(unsigned int u) {unitSize = u;} + + void registerLayer(float, sol::object); + void registerDecoLayer(float, sol::object); }; /** |