aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--Assets/world/world1/layers/0/forestTileBack.pngbin0 -> 27599 bytes
-rw-r--r--Assets/world/world1/layers/0/hitbox.pngbin0 -> 4782 bytes
-rw-r--r--Assets/world/world1/layers/0/normal.pngbin0 -> 4196 bytes
-rw-r--r--Assets/world/world1/layers/0/texture.pngbin0 -> 6142 bytes
-rw-r--r--Assets/world/world1/layers/1/hitbox.pngbin0 -> 3972 bytes
-rw-r--r--Assets/world/world1/layers/1/normal.pngbin0 -> 4196 bytes
-rw-r--r--Assets/world/world1/layers/1/texture.pngbin0 -> 4573 bytes
-rw-r--r--Assets/world/world1/layers/deco/forestTileBack.pngbin0 -> 19045 bytes
-rw-r--r--Assets/world/world1/layers/deco/forestTileFront.pngbin0 -> 15814 bytes
-rw-r--r--Assets/world/world1/layers/deco/forestTileMid.pngbin0 -> 18365 bytes
-rw-r--r--Assets/world/world1/layers/deco/normal.pngbin0 -> 146 bytes
-rw-r--r--Assets/world/world1/layers/deco/texture.pngbin0 -> 7285 bytes
-rw-r--r--Makefile1
-rw-r--r--Scripts/init.lua37
-rw-r--r--Scripts/world.lua121
m---------lib/sol20
-rw-r--r--src/components/Component.hpp4
-rw-r--r--src/components/Physics.hpp30
-rw-r--r--src/components/Position.hpp25
-rw-r--r--src/components/Render.hpp19
-rw-r--r--src/components/Velocity.hpp21
-rw-r--r--src/engine.cpp31
-rw-r--r--src/events/render.hpp36
-rw-r--r--src/events/world.hpp12
-rw-r--r--src/input.hpp22
-rw-r--r--src/physics.cpp52
-rw-r--r--src/render.cpp73
-rw-r--r--src/render.hpp17
-rw-r--r--src/script.cpp22
-rw-r--r--src/script/vectors.cpp108
-rw-r--r--src/script/vectors.hpp31
-rw-r--r--src/text.cpp38
-rw-r--r--src/text.hpp21
-rw-r--r--src/texture.cpp24
-rw-r--r--src/texture.hpp2
-rw-r--r--src/ui.cpp56
-rw-r--r--src/ui.hpp42
-rw-r--r--src/world.cpp321
-rw-r--r--src/world.hpp109
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
new file mode 100644
index 0000000..f1c1a8b
--- /dev/null
+++ b/Assets/world/world1/layers/0/forestTileBack.png
Binary files differ
diff --git a/Assets/world/world1/layers/0/hitbox.png b/Assets/world/world1/layers/0/hitbox.png
new file mode 100644
index 0000000..9ae9622
--- /dev/null
+++ b/Assets/world/world1/layers/0/hitbox.png
Binary files differ
diff --git a/Assets/world/world1/layers/0/normal.png b/Assets/world/world1/layers/0/normal.png
new file mode 100644
index 0000000..bc2ede6
--- /dev/null
+++ b/Assets/world/world1/layers/0/normal.png
Binary files differ
diff --git a/Assets/world/world1/layers/0/texture.png b/Assets/world/world1/layers/0/texture.png
new file mode 100644
index 0000000..429107e
--- /dev/null
+++ b/Assets/world/world1/layers/0/texture.png
Binary files differ
diff --git a/Assets/world/world1/layers/1/hitbox.png b/Assets/world/world1/layers/1/hitbox.png
new file mode 100644
index 0000000..0a9a37e
--- /dev/null
+++ b/Assets/world/world1/layers/1/hitbox.png
Binary files differ
diff --git a/Assets/world/world1/layers/1/normal.png b/Assets/world/world1/layers/1/normal.png
new file mode 100644
index 0000000..bc2ede6
--- /dev/null
+++ b/Assets/world/world1/layers/1/normal.png
Binary files differ
diff --git a/Assets/world/world1/layers/1/texture.png b/Assets/world/world1/layers/1/texture.png
new file mode 100644
index 0000000..149273f
--- /dev/null
+++ b/Assets/world/world1/layers/1/texture.png
Binary files differ
diff --git a/Assets/world/world1/layers/deco/forestTileBack.png b/Assets/world/world1/layers/deco/forestTileBack.png
new file mode 100644
index 0000000..c5e9dac
--- /dev/null
+++ b/Assets/world/world1/layers/deco/forestTileBack.png
Binary files differ
diff --git a/Assets/world/world1/layers/deco/forestTileFront.png b/Assets/world/world1/layers/deco/forestTileFront.png
new file mode 100644
index 0000000..234520c
--- /dev/null
+++ b/Assets/world/world1/layers/deco/forestTileFront.png
Binary files differ
diff --git a/Assets/world/world1/layers/deco/forestTileMid.png b/Assets/world/world1/layers/deco/forestTileMid.png
new file mode 100644
index 0000000..b51d0bc
--- /dev/null
+++ b/Assets/world/world1/layers/deco/forestTileMid.png
Binary files differ
diff --git a/Assets/world/world1/layers/deco/normal.png b/Assets/world/world1/layers/deco/normal.png
new file mode 100644
index 0000000..767f7e3
--- /dev/null
+++ b/Assets/world/world1/layers/deco/normal.png
Binary files differ
diff --git a/Assets/world/world1/layers/deco/texture.png b/Assets/world/world1/layers/deco/texture.png
new file mode 100644
index 0000000..ce37a03
--- /dev/null
+++ b/Assets/world/world1/layers/deco/texture.png
Binary files differ
diff --git a/Makefile b/Makefile
index a14fb2b..4ec54da 100644
--- a/Makefile
+++ b/Makefile
@@ -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);
};
/**