diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/components/Light.hpp | 54 | ||||
-rw-r--r-- | src/components/Render.hpp | 11 | ||||
-rw-r--r-- | src/components/Script.hpp | 15 | ||||
-rw-r--r-- | src/engine.cpp | 13 | ||||
-rw-r--r-- | src/render.cpp | 105 | ||||
-rw-r--r-- | src/render.hpp | 4 | ||||
-rw-r--r-- | src/script.cpp | 22 | ||||
-rw-r--r-- | src/texture.cpp | 20 | ||||
-rw-r--r-- | src/texture.hpp | 38 |
9 files changed, 261 insertions, 21 deletions
diff --git a/src/components/Light.hpp b/src/components/Light.hpp new file mode 100644 index 0000000..ee215a6 --- /dev/null +++ b/src/components/Light.hpp @@ -0,0 +1,54 @@ +/** + * @file Light.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 COMPONENT_LIGHT_HPP_ +#define COMPONENT_LIGHT_HPP_ + +#include "Component.hpp" + +struct Light : Component<Light>, entityx::Component<Light> +{ +public: + float r, g, b; + float strength; + + Light() {} + Light(float _r, float _g, float _z, float _s) : + r(_r), g(_g), b(_z), strength(_s) {} + + Light FromLua(sol::object ref) + { + if (ref.get_type() == sol::type::table) { + sol::table tab = ref; + if (tab["r"] == sol::type::number) + this->r = tab["r"]; + if (tab["g"] == sol::type::number) + this->g = tab["g"]; + if (tab["b"] == sol::type::number) + this->b = tab["b"]; + if (tab["strength"] == sol::type::number) + this->strength = tab["strength"]; + } else { + throw std::string("Light component must be a table"); + } + return *this; + } +}; + +#endif//COMPONENT_LIGHT_HPP_ diff --git a/src/components/Render.hpp b/src/components/Render.hpp index 451f2d1..3f1750f 100644 --- a/src/components/Render.hpp +++ b/src/components/Render.hpp @@ -19,12 +19,15 @@ #define COMPONENT_RENDER_HPP_ #include "Component.hpp" +#include "texture.hpp" struct Render : Component<Render>, entityx::Component<Render> { public: - std::string texture; + Texture texture; + Texture normal; bool visible; + bool flipX = false; Render(std::string _file) : texture(_file), visible(true) {} @@ -38,7 +41,11 @@ public: if (tab["visible"].get_type() == sol::type::boolean) this->visible = tab["visible"]; if (tab["texture"].get_type() == sol::type::string) - this->texture = tab["texture"]; + this->texture = Texture(static_cast<std::string>(tab["texture"])); + if (tab["normal"].get_type() == sol::type::string) + this->normal = Texture(static_cast<std::string>(tab["normal"])); + if (tab["flipx"].get_type() == sol::type::boolean) + this->flipX = tab["flipx"]; } else { throw std::string( "Render component table formatted incorrectly" diff --git a/src/components/Script.hpp b/src/components/Script.hpp index 66addc8..b3c89f3 100644 --- a/src/components/Script.hpp +++ b/src/components/Script.hpp @@ -42,11 +42,24 @@ public: return *this; } - void exec(void) { + void exec(void) + { if (caller["Idle"] == sol::type::function) caller["Idle"](caller); // Call idle function and pass itself // in or to fulfill the 'self' param } + + void updatePhysics(void) + { + if (caller["PhysicsIdle"] == sol::type::function) + caller["PhysicsIdle"](caller); + } + + void updateRender(void) + { + if (caller["RenderIdle"] == sol::type::function) + caller["RenderIdle"](caller); + } }; #endif // COMPONENT_SCRIPT_HPP_ diff --git a/src/engine.cpp b/src/engine.cpp index f235651..2916a6e 100644 --- a/src/engine.cpp +++ b/src/engine.cpp @@ -30,6 +30,10 @@ #include "components/Position.hpp" #include "components/Velocity.hpp" +using namespace std::chrono_literals; +namespace cr = std::chrono; +typedef std::chrono::high_resolution_clock mc; + int Engine::init(void) { systems.add<GameRunSystem>(); @@ -46,10 +50,6 @@ int Engine::init(void) void Engine::logicLoop(void) { - using namespace std::chrono_literals; - namespace cr = std::chrono; - typedef std::chrono::high_resolution_clock mc; - entityx::TimeDelta dt = 0; /**< Elapsed milliseconds since each loop */ double elapsed = 0; @@ -67,6 +67,7 @@ void Engine::logicLoop(void) }); systems.update<InputSystem>(dt); + systems.update<ScriptSystem>(dt); /******************* * LOGIC UPDATES * @@ -97,9 +98,9 @@ void Engine::logicLoop(void) void Engine::renderLoop(void) { + entityx::TimeDelta dt = 0; /**< Elapsed milliseconds since each loop */ while (shouldRun()) { - systems.update<RenderSystem>(0); - std::this_thread::yield(); + systems.update<RenderSystem>(dt); } } diff --git a/src/render.cpp b/src/render.cpp index 2771535..7ae2fd7 100644 --- a/src/render.cpp +++ b/src/render.cpp @@ -21,6 +21,8 @@ #include <render.hpp> #include <components/Render.hpp> #include <components/Position.hpp> +#include <components/Light.hpp> +#include <components/Script.hpp> void RenderSystem::configure([[maybe_unused]] entityx::EntityManager& entities, [[maybe_unused]] entityx::EventManager& events) @@ -32,11 +34,18 @@ void RenderSystem::update([[maybe_unused]] entityx::EntityManager& entities, [[maybe_unused]] entityx::EventManager& events, [[maybe_unused]] entityx::TimeDelta dt) { + // TODO move these to only happen once to speed up rendering GLuint s = worldShader.getProgram(); GLuint v = worldShader.getUniform("view"); GLuint p = worldShader.getUniform("projection"); GLuint m = worldShader.getUniform("model"); GLuint a = worldShader.getAttribute("vertex"); + GLuint t = worldShader.getAttribute("texc"); + + GLuint q = worldShader.getUniform("textu"); + GLuint n = worldShader.getUniform("normu"); + GLuint b = worldShader.getUniform("AmbientLight"); + GLuint f = worldShader.getUniform("Flipped"); /*********** * SETUP * @@ -60,6 +69,7 @@ void RenderSystem::update([[maybe_unused]] entityx::EntityManager& entities, ); glm::mat4 model = glm::mat4(1.0f); + model = glm::scale(model, glm::vec3(2.0f)); glUseProgram(s); @@ -74,31 +84,99 @@ void RenderSystem::update([[maybe_unused]] entityx::EntityManager& entities, glEnable(GL_POLYGON_OFFSET_FILL); glEnableVertexAttribArray(a); + glEnableVertexAttribArray(t); + + // Ambient light, for now this is static + GLfloat amb[4] = {1.0f, 1.0f, 1.0f, 0.0f}; + glUniform4fv(b, 1, amb); + + /************** + * LIGHTING * + **************/ + + + std::vector<glm::vec3> lightPos; + std::vector<glm::vec4> lightColor; + int lightNum = 0; + + entities.each<Light, Position>([&] + (entityx::Entity, Light &l, Position &p){ + + lightPos.push_back(glm::vec3(p.x, p.y,-10.0)); + lightColor.push_back(glm::vec4(l.r, l.g, l.b, l.strength)); + lightNum++; + }); + + glUniform1i(worldShader.getUniform("LightNum"), lightNum); + glUniform3fv(worldShader.getUniform("LightPos"), + lightPos.size(), + reinterpret_cast<GLfloat*>(lightPos.data())); + glUniform4fv(worldShader.getUniform("LightColor"), + lightColor.size(), + reinterpret_cast<GLfloat*>(lightColor.data())); /************* * DRAWING * *************/ entities.each<Render, Position>( - [this, a](entityx::Entity, Render &r, Position &p) { + [this, a, q, t, n, f](entityx::Entity, Render &r, Position &p) { if (!r.visible) return; + // If our component was created via script, call the entity's + // RenderIdle function + //if (e.has_component<Scripted>()) { + // e.component<Scripted>()->updateRender(); + //} + + float w = r.texture.width/2.0f; + float h = r.texture.height; + GLuint tri_vbo; GLfloat tri_data[] = { - (float)p.x-10.0f, (float)p.y-10.0f, 00.0f, - (float)p.x+10.0f, (float)p.y-10.0f, 00.0f, - (float)p.x+00.0f, (float)p.y+10.0f, 00.0f, + (float)p.x-w, (float)p.y , 00.0f, 0.0f, 1.0f, + (float)p.x+w, (float)p.y , 00.0f, 1.0f, 1.0f, + (float)p.x-w, (float)p.y+h, 00.0f, 0.0f, 0.0f, + + (float)p.x+w, (float)p.y , 00.0f, 1.0f, 1.0f, + (float)p.x+w, (float)p.y+h, 00.0f, 1.0f, 0.0f, + (float)p.x-w, (float)p.y+h, 00.0f, 0.0f, 0.0f, }; + bool flipped = false; + + // TODO flip nicely (aka model transformations) + if (r.flipX) { + std::swap(tri_data[3], tri_data[8]); + tri_data[13] = tri_data[3]; + + std::swap(tri_data[23], tri_data[28]); + tri_data[18] = tri_data[23]; + + flipped = true; + } + + glUniform1i(f, flipped ? 1 : 0); + + glActiveTexture(GL_TEXTURE0); + glBindTexture(GL_TEXTURE_2D, r.texture.tex); + glUniform1i(q, 0); + + glActiveTexture(GL_TEXTURE1); + glBindTexture(GL_TEXTURE_2D, r.normal.tex); + glUniform1i(n, 1); + glGenBuffers(1, &tri_vbo); glBindBuffer(GL_ARRAY_BUFFER, tri_vbo); glBufferData(GL_ARRAY_BUFFER, sizeof(tri_data), tri_data, GL_STREAM_DRAW); - glVertexAttribPointer(a, 3, GL_FLOAT, GL_FALSE, 0, 0); - glDrawArrays(GL_TRIANGLES, 0, 3); - + glVertexAttribPointer(a, 3, GL_FLOAT, GL_FALSE, + 5*sizeof(float), 0); + glVertexAttribPointer(t, 2, GL_FLOAT, GL_FALSE, + 5*sizeof(float), (void*)(3*sizeof(float))); + glDrawArrays(GL_TRIANGLES, 0, 6); }); @@ -106,6 +184,7 @@ void RenderSystem::update([[maybe_unused]] entityx::EntityManager& entities, * CLEANUP * *************/ glDisableVertexAttribArray(a); + glDisableVertexAttribArray(t); glDisable(GL_POLYGON_OFFSET_FILL); glDisable(GL_CULL_FACE); @@ -163,6 +242,16 @@ int RenderSystem::init(void) worldShader.addUniform("model"); worldShader.addAttribute("vertex"); + worldShader.addAttribute("texc"); + + worldShader.addUniform("textu"); + worldShader.addUniform("normu"); + + worldShader.addUniform("LightPos"); + worldShader.addUniform("LightColor"); + worldShader.addUniform("LightNum"); + worldShader.addUniform("AmbientLight"); + worldShader.addUniform("Flipped"); glEnableVertexAttribArray(worldShader.getAttribute("vertex")); glUseProgram(worldShader.getProgram()); @@ -170,7 +259,7 @@ int RenderSystem::init(void) // TODO //glPolygonOffset(1.0, 1.0); - glClearColor(0.6, 0.8, 1.0, 0.0); + //glClearColor(0.6, 0.8, 1.0, 0.0); return 0; } diff --git a/src/render.hpp b/src/render.hpp index 6362d63..5d3025c 100644 --- a/src/render.hpp +++ b/src/render.hpp @@ -38,8 +38,8 @@ class RenderSystem : public entityx::System<RenderSystem> { private: constexpr static const char *title = "gamedev2"; - constexpr static int width = 640; - constexpr static int height = 480; + constexpr static int width = 1280; + constexpr static int height = 720; std::unique_ptr<SDL_Window, void (*)(SDL_Window *)> window; SDL_GLContext context; diff --git a/src/script.cpp b/src/script.cpp index 80ac538..fa485bc 100644 --- a/src/script.cpp +++ b/src/script.cpp @@ -36,10 +36,14 @@ void ScriptSystem::configure(entityx::EntityManager& entities, //init(); } -void ScriptSystem::update([[maybe_unused]] entityx::EntityManager& entites, +#include <components/Script.hpp> +void ScriptSystem::update([[maybe_unused]] entityx::EntityManager& entities, [[maybe_unused]] entityx::EventManager& events, [[maybe_unused]] entityx::TimeDelta dt) { + entities.each<Scripted>([](entityx::Entity, Scripted &s){ + s.updatePhysics(); + }); } @@ -80,6 +84,7 @@ void ScriptSystem::doFile(void) #include <components/Render.hpp> #include <components/Script.hpp> #include <components/Velocity.hpp> +#include <components/Light.hpp> void ScriptSystem::scriptExport(void) { @@ -98,7 +103,8 @@ void ScriptSystem::scriptExport(void) lua.new_usertype<Render>("Render", sol::constructors<Render(std::string), Render()>(), "visible", &Render::visible, - "texture", &Render::texture); + "texture", &Render::texture, + "flipx", &Render::flipX); lua.new_usertype<Velocity>("Velocity", sol::constructors<Velocity(double, double), Velocity()>(), @@ -108,6 +114,13 @@ void ScriptSystem::scriptExport(void) lua.new_usertype<Player>("Player", sol::constructors<Player(void), Player()>()); + lua.new_usertype<Light>("Light", + sol::constructors<Light(float, float, float, float)>(), + "r", &Light::r, + "g", &Light::g, + "b", &Light::b, + "strength", &Light::strength); + auto gamespace = lua["game"].get_or_create<sol::table>(); gamespace.set_function("spawn", func); } @@ -162,6 +175,11 @@ sol::table ScriptSystem::spawn(sol::object param) (*toRet)["Player"] = e.assign<Player>().get(); } + if (tab["Light"] != nullptr) { + (*toRet)["Light"] = + e.assign<Light>(Light().FromLua(tab["Light"])).get(); + } + } else { // TODO better logging std::cerr << "Parameter to spawn() must be a table!" << std::endl; diff --git a/src/texture.cpp b/src/texture.cpp new file mode 100644 index 0000000..5604812 --- /dev/null +++ b/src/texture.cpp @@ -0,0 +1,20 @@ +#include "texture.hpp" + +Texture::Texture(std::string filename) +{ + unsigned char* image = SOIL_load_image(filename.c_str(), + &width, &height, 0, + SOIL_LOAD_RGBA); + + glGenTextures(1, &tex); // Turns "object" into a texture + glBindTexture(GL_TEXTURE_2D, tex); // Binds "object" to the top of the stack + glPixelStoref(GL_UNPACK_ALIGNMENT, 1); + glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST); // Sets the "min" filter + glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST); // The the "max" filter of the stack + glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); // Wrap the texture to the matrix + glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); // + glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, width, height, 0, + GL_RGBA, GL_UNSIGNED_BYTE, image); + + SOIL_free_image_data(image); +} diff --git a/src/texture.hpp b/src/texture.hpp new file mode 100644 index 0000000..16987f8 --- /dev/null +++ b/src/texture.hpp @@ -0,0 +1,38 @@ +/** + * @file texture.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 TEXTURE_HPP_ +#define TEXTURE_HPP_ + +#include <soil/SOIL.h> +#include <SDL2/SDL_opengl.h> +#include <string> + +class Texture +{ +private: +public: + GLuint tex; + int width; + int height; + Texture() {}; + Texture(std::string); +}; + +#endif//TEXTURE_HPP_ |