/** * @file world.hpp * Manages the world systems * * Copyright (C) 2019 Belle-Isle, Andrew * * 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 . */ #ifndef SYSTEM_WORLD_HPP_ #define SYSTEM_WORLD_HPP_ #include #include #include #include #include "texture.hpp" #include "events/render.hpp" #include // For entity position #include // For entity velocity #include // For entity hitbox(es) struct WorldMaterial { bool passable = false; Texture texture; Texture normal; WorldMaterial(sol::table tab) { if (tab["texture"] != nullptr) { sol::object t = tab["texture"]; texture = Texture(t); } if (tab["normal"] != nullptr) { sol::object n = tab["normal"]; normal = Texture(n); } if (tab["passable"] == sol::type::boolean) { passable = tab["passable"]; } } }; 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> 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(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; private: unsigned int seed; unsigned int layers; unsigned int unitSize; std::vector> solidLayers; std::vector> drawLayers; std::vector getIntersectingPlanes(glm::vec3 origin, Physics &phys); bool isSolid(glm::vec3 pos); protected: // RENDER std::vector meshAdd; GLuint worldVBO; public: /* VARS */ sol::function generate; sol::function registerMat; World() {} World(sol::object ref); ~World() { registerMat = sol::nil; generate = sol::nil; } std::tuple getSize(); /* RENDERING */ void generateMesh(); /* SEED */ unsigned int getSeed(); unsigned int setSeed(unsigned int); /* 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); }; /** * @class WorldSystem * Handles the game's world system */ class WorldSystem : public entityx::System { private: std::vector worlds; World* currentWorld; public: WorldSystem(void): currentWorld(nullptr) {} ~WorldSystem(void) { currentWorld = nullptr; worlds.clear(); } World* addWorld(sol::object); World* current() {return currentWorld;}; void cleanup() { worlds.clear(); } /** * Prepares the system for running. */ void configure(entityx::EntityManager& entities, entityx::EventManager& events) final; /** * Updates the world ticks (entity spawns and world events) */ void update(entityx::EntityManager& entites, entityx::EventManager& events, entityx::TimeDelta dt) final; }; #endif // SYSTEM_WORLD_HPP_