mirror of
https://github.com/NoCheatPlus/NoCheatPlus.git
synced 2025-01-12 18:50:50 +01:00
[BLEEDING] Fixes for passable/rt preventing moving out of blocks.
Passable/ray-tracing might disallow players moving out of blocks which they are stuck in on some occasions. In order to allow moving out of a block, ray tracing is repeated with the first block ignored for some cases.
This commit is contained in:
parent
ee0ca14cef
commit
d7ca742c57
@ -26,19 +26,36 @@ public class Passable extends Check {
|
|||||||
{
|
{
|
||||||
// Simple check (only from, to, player.getLocation).
|
// Simple check (only from, to, player.getLocation).
|
||||||
|
|
||||||
// TODO: account for actual bounding box.
|
// TODO: Future: Account for the players bounding box? [test very-strict setting for at least the end points...]
|
||||||
String tags = "";
|
String tags = "";
|
||||||
|
// Block distances (sum, max) for from-to (not for loc!).
|
||||||
|
final int manhattan = from.manhattan(to);
|
||||||
boolean toPassable = to.isPassable();
|
boolean toPassable = to.isPassable();
|
||||||
// TODO: Config settings, extra flag for further processing.
|
// General condition check for using ray-tracing.
|
||||||
if (toPassable && cc.passableRayTracingCheck && (!cc.passableRayTracingVclipOnly || from.getY() > to.getY()) && (!cc.passableRayTracingBlockChangeOnly || from.getBlockX() != to.getBlockX() || from.getBlockY() != to.getBlockY() || from.getBlockZ() != to.getBlockZ())){
|
if (toPassable && cc.passableRayTracingCheck && (!cc.passableRayTracingVclipOnly || from.getY() > to.getY()) && (!cc.passableRayTracingBlockChangeOnly || manhattan > 0)){
|
||||||
rayTracing.set(from, to);
|
rayTracing.set(from, to);
|
||||||
rayTracing.loop();
|
rayTracing.loop();
|
||||||
if (rayTracing.collides() || rayTracing.getStepsDone() >= rayTracing.getMaxSteps()){
|
if (rayTracing.collides() || rayTracing.getStepsDone() >= rayTracing.getMaxSteps()){
|
||||||
toPassable = false;
|
final int maxBlockDist = manhattan <= 1 ? manhattan : from.maxBlockDist(to);
|
||||||
tags = "raytracing";
|
if (maxBlockDist <= 1 && rayTracing.getStepsDone() == 1 && !from.isPassable()){
|
||||||
|
// Redo ray-tracing for moving out of blocks.
|
||||||
|
rayTracing.set(from, to);
|
||||||
|
rayTracing.setIgnorefirst();
|
||||||
|
rayTracing.loop();
|
||||||
|
if (rayTracing.collides() || rayTracing.getStepsDone() >= rayTracing.getMaxSteps()){
|
||||||
|
toPassable = false;
|
||||||
|
tags = "raytracing_2x_";
|
||||||
|
}
|
||||||
|
else if (cc.debug){
|
||||||
|
System.out.println(player.getName() + " passable: allow moving out of a block.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else{
|
||||||
|
toPassable = false;
|
||||||
|
tags = "raytracing_";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// TODO: If accuracy is set, also check the head position (or bounding box right away).
|
// TODO: Future: If accuracy is demanded, also check the head position (or bounding box right away).
|
||||||
rayTracing.cleanup();
|
rayTracing.cleanup();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -53,7 +70,6 @@ public class Passable extends Check {
|
|||||||
|
|
||||||
// Check the players location if different from others.
|
// Check the players location if different from others.
|
||||||
// (It provides a better set-back for some exploits.)
|
// (It provides a better set-back for some exploits.)
|
||||||
// Location loc = player.getLocation();
|
|
||||||
final int lbX = loc.getBlockX();
|
final int lbX = loc.getBlockX();
|
||||||
final int lbY = loc.getBlockY();
|
final int lbY = loc.getBlockY();
|
||||||
final int lbZ = loc.getBlockZ();
|
final int lbZ = loc.getBlockZ();
|
||||||
@ -62,25 +78,27 @@ public class Passable extends Check {
|
|||||||
if (from.isPassable()){
|
if (from.isPassable()){
|
||||||
// From should be the set-back.
|
// From should be the set-back.
|
||||||
loc = null;
|
loc = null;
|
||||||
tags = "into";
|
tags += "into";
|
||||||
} else if (BlockProperties.isPassable(from.getBlockCache(), loc.getX(), loc.getY(), loc.getZ(), from.getTypeId(lbX, lbY, lbZ))){
|
} else if (BlockProperties.isPassable(from.getBlockCache(), loc.getX(), loc.getY(), loc.getZ(), from.getTypeId(lbX, lbY, lbZ))){
|
||||||
|
tags += "into_shift";
|
||||||
|
}
|
||||||
// } else if (BlockProperties.isPassableExact(from.getBlockCache(), loc.getX(), loc.getY(), loc.getZ(), from.getTypeId(lbX, lbY, lbZ))){
|
// } else if (BlockProperties.isPassableExact(from.getBlockCache(), loc.getX(), loc.getY(), loc.getZ(), from.getTypeId(lbX, lbY, lbZ))){
|
||||||
// (Mind that this can be the case on the same block theoretically.)
|
// (Mind that this can be the case on the same block theoretically.)
|
||||||
// Keep loc as set-back.
|
// Keep loc as set-back.
|
||||||
}
|
// }
|
||||||
else if (!from.isSameBlock(lbX, lbY, lbZ)){
|
else if (!from.isSameBlock(lbX, lbY, lbZ)){
|
||||||
// Otherwise keep loc as set-back.
|
// Otherwise keep loc as set-back.
|
||||||
tags = "cross_shift";
|
tags += "cross_shift";
|
||||||
}
|
}
|
||||||
else if (to.isBlockAbove(from) && BlockProperties.isPassable(from.getBlockCache(), from.getX(), from.getY() + player.getEyeHeight(), from.getZ(), from.getTypeId(from.getBlockX(), Location.locToBlock(from.getY() + player.getEyeHeight()), from.getBlockZ()))){
|
else if (manhattan == 1 && to.isBlockAbove(from) && BlockProperties.isPassable(from.getBlockCache(), from.getX(), from.getY() + player.getEyeHeight(), from.getZ(), from.getTypeId(from.getBlockX(), Location.locToBlock(from.getY() + player.getEyeHeight()), from.getBlockZ()))){
|
||||||
// else if (to.isBlockAbove(from) && BlockProperties.isPassableExact(from.getBlockCache(), from.getX(), from.getY() + player.getEyeHeight(), from.getZ(), from.getTypeId(from.getBlockX(), Location.locToBlock(from.getY() + player.getEyeHeight()), from.getBlockZ()))){
|
// else if (to.isBlockAbove(from) && BlockProperties.isPassableExact(from.getBlockCache(), from.getX(), from.getY() + player.getEyeHeight(), from.getZ(), from.getTypeId(from.getBlockX(), Location.locToBlock(from.getY() + player.getEyeHeight()), from.getBlockZ()))){
|
||||||
// Allow the move up if the head is free.
|
// Allow the move up if the head is free.
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
else if (!from.isSameBlock(to)){
|
else if (manhattan > 0){
|
||||||
// Otherwise keep from as set-back.
|
// Otherwise keep from as set-back.
|
||||||
loc = null;
|
loc = null;
|
||||||
tags = "cross";
|
tags += "cross";
|
||||||
}
|
}
|
||||||
else{
|
else{
|
||||||
// All blocks are the same, allow the move.
|
// All blocks are the same, allow the move.
|
||||||
|
@ -5,6 +5,8 @@ public class PassableRayTracing extends RayTracing{
|
|||||||
protected BlockCache blockCache = null;
|
protected BlockCache blockCache = null;
|
||||||
|
|
||||||
protected boolean collides = false;
|
protected boolean collides = false;
|
||||||
|
|
||||||
|
protected boolean ignorefirst = false;
|
||||||
|
|
||||||
public BlockCache getBlockCache() {
|
public BlockCache getBlockCache() {
|
||||||
return blockCache;
|
return blockCache;
|
||||||
@ -31,12 +33,27 @@ public class PassableRayTracing extends RayTracing{
|
|||||||
public void set(double x0, double y0, double z0, double x1, double y1, double z1) {
|
public void set(double x0, double y0, double z0, double x1, double y1, double z1) {
|
||||||
super.set(x0, y0, z0, x1, y1, z1);
|
super.set(x0, y0, z0, x1, y1, z1);
|
||||||
collides = false;
|
collides = false;
|
||||||
|
ignorefirst = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean collides(){
|
public boolean collides(){
|
||||||
return collides;
|
return collides;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Ignore the first block. Must be called after set, because set will override this with false.
|
||||||
|
*/
|
||||||
|
public void setIgnorefirst(){
|
||||||
|
this.ignorefirst = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test if the first block is set to be ignored (resets to false with set).
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public boolean getIgnoreFirst(){
|
||||||
|
return ignorefirst;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Remove reference to BlockCache.
|
* Remove reference to BlockCache.
|
||||||
@ -50,6 +67,9 @@ public class PassableRayTracing extends RayTracing{
|
|||||||
@Override
|
@Override
|
||||||
protected boolean step(final int blockX, final int blockY, final int blockZ, final double oX, final double oY, final double oZ, final double dT) {
|
protected boolean step(final int blockX, final int blockY, final int blockZ, final double oX, final double oY, final double oZ, final double dT) {
|
||||||
// Just delegate.
|
// Just delegate.
|
||||||
|
if (step == 1 && ignorefirst){
|
||||||
|
return true;
|
||||||
|
}
|
||||||
if (BlockProperties.isPassableRay(blockCache, blockX, blockY, blockZ, oX, oY, oZ, dX, dY, dZ, dT)){
|
if (BlockProperties.isPassableRay(blockCache, blockX, blockY, blockZ, oX, oY, oZ, dX, dY, dZ, dT)){
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -268,6 +268,26 @@ public class PlayerLocation {
|
|||||||
public boolean isSamePos(final Location loc) {
|
public boolean isSamePos(final Location loc) {
|
||||||
return x == loc.getX() && z == loc.getZ() && y == loc.getY();
|
return x == loc.getX() && z == loc.getZ() && y == loc.getY();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Manhattan distance, see Trigutil.
|
||||||
|
* @param other
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public int manhattan(final PlayerLocation other){
|
||||||
|
// TODO: Consider using direct field access from other methods as well.
|
||||||
|
return TrigUtil.manhattan(this.blockX, this.blockY, this.blockZ, other.blockX, other.blockY, other.blockZ);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Maximum block distance comparing dx, dy, dz.
|
||||||
|
* @param other
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public int maxBlockDist(final PlayerLocation other){
|
||||||
|
// TODO: Consider using direct field access from other methods as well.
|
||||||
|
return TrigUtil.maxDistance(this.blockX, this.blockY, this.blockZ, other.blockX, other.blockY, other.blockZ);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks if the player is above stairs.
|
* Checks if the player is above stairs.
|
||||||
|
@ -331,5 +331,33 @@ public class TrigUtil {
|
|||||||
else if (yawDiff > 180f) yawDiff -= 360f;
|
else if (yawDiff > 180f) yawDiff -= 360f;
|
||||||
return yawDiff;
|
return yawDiff;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Manhattan distance (steps along the sides of an orthogonal grid).
|
||||||
|
* @param x1
|
||||||
|
* @param y1
|
||||||
|
* @param z1
|
||||||
|
* @param x2
|
||||||
|
* @param y2
|
||||||
|
* @param z2
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public static int manhattan(final int x1, final int y1, final int z1, final int x2, final int y2, final int z2){
|
||||||
|
return Math.abs(x1 - x2) + Math.abs(y1 - y2) + Math.abs(z1 - z2);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Maximum distance comparing dx, dy, dz.
|
||||||
|
* @param x1
|
||||||
|
* @param y1
|
||||||
|
* @param z1
|
||||||
|
* @param x2
|
||||||
|
* @param y2
|
||||||
|
* @param z2
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public static int maxDistance(final int x1, final int y1, final int z1, final int x2, final int y2, final int z2){
|
||||||
|
return Math.max(Math.max(Math.abs(x1 - x2), Math.abs(y1 - y2)), Math.abs(z1 - z2));
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user