diff options
Diffstat (limited to 'src/world.cpp')
-rw-r--r-- | src/world.cpp | 325 |
1 files changed, 186 insertions, 139 deletions
diff --git a/src/world.cpp b/src/world.cpp index cd89a22..48e54e3 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,160 @@ 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 (inter.size()) { + p.standing = true; + } + } + + // 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; +} + - return Y; +/********* +* 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(); +} + +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 +298,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(); } |