From 5ae26c44e1f5ec52b7025ba4d76fffea3f592894 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?N=C3=A9meth=20Noel?= Date: Wed, 7 Jul 2021 17:56:17 +0200 Subject: [PATCH 1/3] Initial npe fix --- .../server/collision/CollisionUtils.java | 45 +++++++------------ .../net/minestom/server/entity/Entity.java | 32 +++++++------ 2 files changed, 33 insertions(+), 44 deletions(-) diff --git a/src/main/java/net/minestom/server/collision/CollisionUtils.java b/src/main/java/net/minestom/server/collision/CollisionUtils.java index b05cb13af..6b2fcf87c 100644 --- a/src/main/java/net/minestom/server/collision/CollisionUtils.java +++ b/src/main/java/net/minestom/server/collision/CollisionUtils.java @@ -69,16 +69,12 @@ public class CollisionUtils { // used to determine if 'remainingLength' should be used boolean collisionFound = false; for (int i = 0; i < Math.abs(blockLength); i++) { - final OneStepResult oneStepResult = stepOnce(instance, originChunk, axis, sign, corners); - corners = oneStepResult.newCorners; - if (collisionFound = oneStepResult.foundCollision) break; + if (collisionFound = stepOnce(instance, originChunk, axis, sign, corners)) break; } // add remainingLength if (!collisionFound) { - final OneStepResult oneStepResult = stepOnce(instance, originChunk, axis, remainingLength, corners); - corners = oneStepResult.newCorners; - collisionFound = oneStepResult.foundCollision; + collisionFound = stepOnce(instance, originChunk, axis, sign, corners); } // find the corner which moved the least @@ -99,38 +95,37 @@ public class CollisionUtils { * @param instance instance to get blocks from * @param axis the axis to move along * @param corners the corners of the bounding box to consider - * @return the result of one step + * @return true if found collision */ - private static OneStepResult stepOnce(Instance instance, Chunk originChunk, Vec axis, double amount, Vec[] corners) { + private static boolean stepOnce(Instance instance, Chunk originChunk, Vec axis, double amount, Vec[] corners) { final double sign = Math.signum(amount); - Vec[] newCorners = new Vec[corners.length]; for (int cornerIndex = 0; cornerIndex < corners.length; cornerIndex++) { final Vec originalCorner = corners[cornerIndex]; - final Vec corner = originalCorner.add(axis.mul(amount)); + final Vec newCorner = originalCorner.add(axis.mul(amount)); - Chunk chunk = ChunkUtils.retrieve(instance, originChunk, corner); + Chunk chunk = ChunkUtils.retrieve(instance, originChunk, newCorner); if (!ChunkUtils.isLoaded(chunk)) { // Collision at chunk border - return new OneStepResult(corners, true); + return true; } - final Block block = chunk.getBlock(corner); + final Block block = chunk.getBlock(newCorner); // TODO: block collision boxes // TODO: for the moment, always consider a full block if (block.isSolid()) { - newCorners[cornerIndex] = originalCorner.with(((x, y, z) -> new Vec( - Math.abs(axis.x()) > 10e-16 ? originalCorner.blockX() - axis.x() * sign : x, - Math.abs(axis.y()) > 10e-16 ? originalCorner.blockY() - axis.y() * sign : y, - Math.abs(axis.z()) > 10e-16 ? originalCorner.blockZ() - axis.z() * sign : z + corners[cornerIndex] = originalCorner.with(((x, y, z) -> new Vec( + Math.abs(axis.x()) > 10e-16 ? newCorner.blockX() - axis.x() * sign : x, + Math.abs(axis.y()) > 10e-16 ? newCorner.blockY() - axis.y() * sign : y, + Math.abs(axis.z()) > 10e-16 ? newCorner.blockZ() - axis.z() * sign : z ))); - return new OneStepResult(newCorners, true); + return true; } - newCorners[cornerIndex] = corner; + corners[cornerIndex] = newCorner; } - return new OneStepResult(newCorners, false); + return false; } /** @@ -196,14 +191,4 @@ public class CollisionUtils { this.foundCollision = foundCollision; } } - - private static class OneStepResult { - private final Vec[] newCorners; - private final boolean foundCollision; - - public OneStepResult(Vec[] newCorners, boolean foundCollision) { - this.newCorners = newCorners; - this.foundCollision = foundCollision; - } - } } diff --git a/src/main/java/net/minestom/server/entity/Entity.java b/src/main/java/net/minestom/server/entity/Entity.java index 6dde0c1f8..377837463 100644 --- a/src/main/java/net/minestom/server/entity/Entity.java +++ b/src/main/java/net/minestom/server/entity/Entity.java @@ -537,22 +537,25 @@ public class Entity implements Viewable, Tickable, EventHandler, Da // Update velocity if (hasVelocity() || !newVelocity.equals(Vec.ZERO)) { - velocity = newVelocity.mul(tps); // Convert from blocks/tick to blocks/sec - - final Block block = finalChunk.getBlock(position); - final double drag = block.registry().friction(); - if (onGround) { + if (onGround && isNettyClient) { // Stop player velocity - if (isNettyClient) { - velocity = Vec.ZERO; - } - } + velocity = Vec.ZERO; + } else { + final Block block = finalChunk.getBlock(position); + final double drag = block.registry().friction(); - velocity = velocity.with((x, y, z) -> new Vec( - x * drag, - !hasNoGravity() ? y * (1 - gravityDragPerTick) : y, - z * drag - )).with(Vec.Operator.EPSILON); + velocity = newVelocity + // Convert from blocks/tick to blocks/sec + .mul(tps) + // Apply drag + .with((x, y, z) -> new Vec( + x * drag, + !hasNoGravity() ? y * (1 - gravityDragPerTick) : y, + z * drag + )) + // Prevent infinitely decreasing velocity + .with(Vec.Operator.EPSILON); + } } // Synchronization and packets... @@ -1315,6 +1318,7 @@ public class Entity implements Viewable, Tickable, EventHandler, Da * @param newPosition the new position */ private void refreshCoordinate(Point newPosition) { + position = position.withCoord(newPosition); if (hasPassenger()) { for (Entity passenger : getPassengers()) { passenger.refreshCoordinate(newPosition); From 8b35e0db9a091137e2f7bb3ca82981d9470fa58b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?N=C3=A9meth=20Noel?= Date: Wed, 7 Jul 2021 18:07:27 +0200 Subject: [PATCH 2/3] Use remaining length when needed --- src/main/java/net/minestom/server/collision/CollisionUtils.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/main/java/net/minestom/server/collision/CollisionUtils.java b/src/main/java/net/minestom/server/collision/CollisionUtils.java index 6b2fcf87c..ea75723c3 100644 --- a/src/main/java/net/minestom/server/collision/CollisionUtils.java +++ b/src/main/java/net/minestom/server/collision/CollisionUtils.java @@ -74,7 +74,7 @@ public class CollisionUtils { // add remainingLength if (!collisionFound) { - collisionFound = stepOnce(instance, originChunk, axis, sign, corners); + collisionFound = stepOnce(instance, originChunk, axis, remainingLength, corners); } // find the corner which moved the least From ce8e14ede195c1bee8bcc28f6496634b8ddee323 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?N=C3=A9meth=20Noel?= Date: Wed, 7 Jul 2021 18:36:24 +0200 Subject: [PATCH 3/3] Use proper method for refreshing position --- src/main/java/net/minestom/server/entity/Entity.java | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/main/java/net/minestom/server/entity/Entity.java b/src/main/java/net/minestom/server/entity/Entity.java index 377837463..40d6e4d7e 100644 --- a/src/main/java/net/minestom/server/entity/Entity.java +++ b/src/main/java/net/minestom/server/entity/Entity.java @@ -531,7 +531,7 @@ public class Entity implements Viewable, Tickable, EventHandler, Da // Apply the position if changed if (!finalVelocityPosition.samePoint(position)) { - refreshCoordinate(finalVelocityPosition); + refreshPosition((Pos) finalVelocityPosition, true); sendPositionUpdate(true); } @@ -1318,7 +1318,6 @@ public class Entity implements Viewable, Tickable, EventHandler, Da * @param newPosition the new position */ private void refreshCoordinate(Point newPosition) { - position = position.withCoord(newPosition); if (hasPassenger()) { for (Entity passenger : getPassengers()) { passenger.refreshCoordinate(newPosition);