X-Git-Url: http://git.localhorst.tv/?a=blobdiff_plain;f=src%2Fworld%2Fworld.cpp;h=46afc1b2a1525a325273cb776dd4ba64d97891ae;hb=b4e5dad2ed5c7e77573de413f3bf5be88577856d;hp=4f73243c5d4ca61b87b0bd7ba92abca74c834287;hpb=8e9e2bb4b2dd5a4100f4531628ab58002fe253c1;p=blank.git diff --git a/src/world/world.cpp b/src/world/world.cpp index 4f73243..46afc1b 100644 --- a/src/world/world.cpp +++ b/src/world/world.cpp @@ -16,6 +16,7 @@ #include #include #include +#include #include #include #include @@ -51,6 +52,10 @@ Entity::Entity(const Entity &other) noexcept , name(other.name) , bounds(other.bounds) , state(other.state) +, model_transform(1.0f) +, view_transform(1.0f) +, speed(0.0f) +, heading(0.0f, 0.0f, -1.0f) , max_vel(other.max_vel) , max_force(other.max_force) , ref_count(0) @@ -109,13 +114,11 @@ void Entity::SetHead(float p, float y) noexcept { } glm::mat4 Entity::Transform(const glm::ivec3 &reference) const noexcept { - return state.Transform(reference); + return glm::translate(glm::vec3((state.chunk_pos - reference) * Chunk::Extent())) * model_transform; } glm::mat4 Entity::ViewTransform(const glm::ivec3 &reference) const noexcept { - glm::mat4 transform = view_local; - transform[3] += glm::vec4(state.RelativePosition(reference), 0.0f); - return transform; + return Transform(reference) * view_transform; } Ray Entity::Aim(const Chunk::Pos &chunk_offset) const noexcept { @@ -125,43 +128,41 @@ Ray Entity::Aim(const Chunk::Pos &chunk_offset) const noexcept { void Entity::UpdateModel() noexcept { state.AdjustHeading(); + state.orient = glm::quat(glm::vec3(0.0f, state.yaw, 0.0f)); if (model) { - Part::State &body_state = model.BodyState(); - Part::State &eyes_state = model.EyesState(); - if (&body_state != &eyes_state) { - body_state.orientation = glm::quat(glm::vec3(0.0f, state.yaw, 0.0f)); - eyes_state.orientation = glm::quat(glm::vec3(state.pitch, 0.0f, 0.0f)); - } else { - eyes_state.orientation = glm::quat(glm::vec3(state.pitch, state.yaw, 0.0f)); - } + model.EyesState().orientation = glm::quat(glm::vec3(state.pitch, 0.0f, 0.0f)); } } void Entity::Update(float dt) { - UpdateView(); + UpdateTransforms(); UpdateHeading(); if (HasController()) { GetController().Update(*this, dt); } } -void Entity::UpdateView() noexcept { - // create local transform - view_local = Transform(ChunkCoords()); - // clear the translation part - view_local[3] = glm::vec4(0.0f, 0.0f, 0.0f, 1.0f); - // add the model's eyes translation, if any +void Entity::UpdateTransforms() noexcept { + // model transform is the one given by current state + model_transform = state.Transform(state.chunk_pos); + // view transform is either the model's eyes transform or, + // should the entity have no model, the pitch (yaw already is + // in model transform) if (model) { - view_local *= model.EyesTransform(); + view_transform = model.EyesTransform(); + } else { + view_transform = glm::eulerAngleX(state.pitch); } } void Entity::UpdateHeading() noexcept { - if (Moving()) { - heading = normalize(Velocity()); + speed = length(Velocity()); + if (speed > std::numeric_limits::epsilon()) { + heading = Velocity() / speed; } else { - // use -Z (forward axis) of local view transform - heading = -glm::vec3(view_local[2]); + speed = 0.0f; + // use -Z (forward axis) of model transform (our "chest") + heading = -glm::vec3(model_transform[2]); } } @@ -232,7 +233,7 @@ void EntityState::AdjustPosition() noexcept { } void EntityState::AdjustHeading() noexcept { - glm::clamp(pitch, -PI_0p5, PI_0p5); + pitch = glm::clamp(pitch, -PI_0p5, PI_0p5); while (yaw > PI) { yaw -= PI_2p0; } @@ -474,6 +475,15 @@ bool World::Intersection(const Entity &e, const EntityState &s, std::vector &col +) { bool any = false; for (Chunk &cur_chunk : chunks) { if (manhattan_radius(cur_chunk.Position() - reference) > 1) { @@ -488,14 +498,13 @@ bool World::Intersection(const Entity &e, const EntityState &s, std::vectorGetLight(base); + int max_light = 0; + int min_light = 15; + glm::ivec3 acc(0, 0, 0); + for (glm::ivec3 offset(-1, -1, -1); offset.z < 2; ++offset.z) { + for (offset.y = -1; offset.y < 2; ++offset.y) { + for (offset.x = -1; offset.x < 2; ++offset.x) { + BlockLookup block(chunk, base + offset); + if (!block) { + // missing, just ignore it + continue; + } + // otherwise, accumulate the difference times direction + acc += offset * (base_light - block.GetLight()); + max_light = std::max(max_light, block.GetLight()); + min_light = std::min(min_light, block.GetLight()); + } + } + } + dir = acc; + col = glm::vec3(std::pow(0.8f, 15 - max_light)); + amb = glm::vec3(std::pow(0.8f, 15 - min_light)); +} + +namespace { + +PrimitiveMesh::Buffer debug_buf; + +} + +void World::RenderDebug(Viewport &viewport) { + PrimitiveMesh debug_mesh; + PlainColor &prog = viewport.WorldColorProgram(); + for (const Entity &entity : entities) { + debug_buf.OutlineBox(entity.Bounds(), glm::vec4(1.0f, 0.0f, 0.0f, 1.0f)); + debug_mesh.Update(debug_buf); + prog.SetM(entity.Transform(players.front().GetEntity().ChunkCoords())); + debug_mesh.DrawLines(); } }