forked from CaffeineMC/lithium
-
Notifications
You must be signed in to change notification settings - Fork 3
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
fix: add hitboxes inside some voxelshapes of cuboids (e.g. extended p…
…iston base) fixes: Entities within EPSILON of a block will not move backwards fixes: Lithium removes hitboxes inside some blocks. CaffeineMC#60
- Loading branch information
Showing
4 changed files
with
174 additions
and
10 deletions.
There are no files selected for viewing
125 changes: 125 additions & 0 deletions
125
...ain/java/me/jellysquid/mods/lithium/common/shapes/VoxelShapeCuboidWithSurfacesInside.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,125 @@ | ||
package me.jellysquid.mods.lithium.common.shapes; | ||
|
||
import net.minecraft.util.math.AxisCycleDirection; | ||
import net.minecraft.util.math.Box; | ||
import net.minecraft.util.shape.VoxelSet; | ||
import net.minecraft.util.shape.VoxelShape; | ||
|
||
/** | ||
* An efficient implementation of {@link VoxelShape} for a shape with one simple cuboid. | ||
* This is an alternative to VoxelShapeSimpleCube with extra hitboxes inside. | ||
* Vanilla has extra hitboxes at steps of 1/8th or 1/4th of a block depending on the exact coordinates of the shape. | ||
* We are mimicking the effect on collisions here, as otherwise some contraptions would not behave like vanilla. | ||
* @author 2No2Name | ||
*/ | ||
public class VoxelShapeCuboidWithSurfacesInside extends VoxelShapeSimpleCube implements VoxelShapeCaster { | ||
//In bit aligned shapes the bitset adds segments are between minX/Y/Z and maxX/Y/Z. | ||
//Segments all have the same size. There is an additional hitbox between two adjacent segments (if both are inside the shape) | ||
private final double xSegmentSize, ySegmentSize, zSegmentSize; | ||
|
||
public VoxelShapeCuboidWithSurfacesInside(VoxelSet voxels, double minX, double minY, double minZ, double maxX, double maxY, double maxZ, int xRes, int yRes, int zRes) { | ||
super(voxels, minX, minY, minZ, maxX, maxY, maxZ); | ||
//If the voxelshape doesn't contain any extra hitboxes in vanilla on the given axis (only one segment in total) | ||
//we will not have segments inside by using infinity as step size, special case in calculatePenetration | ||
this.xSegmentSize = xRes <= 0 ? Double.POSITIVE_INFINITY : 1D / (1 << xRes); | ||
this.ySegmentSize = yRes <= 0 ? Double.POSITIVE_INFINITY : 1D / (1 << yRes); | ||
this.zSegmentSize = zRes <= 0 ? Double.POSITIVE_INFINITY : 1D / (1 << zRes); | ||
} | ||
|
||
public VoxelShapeCuboidWithSurfacesInside(VoxelSet voxels, double minX, double minY, double minZ, double maxX, double maxY, double maxZ, double xSegmentSize, double ySegmentSize, double zSegmentSize) { | ||
super(voxels, minX, minY, minZ, maxX, maxY, maxZ); | ||
|
||
this.xSegmentSize = xSegmentSize; | ||
this.ySegmentSize = ySegmentSize; | ||
this.zSegmentSize = zSegmentSize; | ||
} | ||
|
||
|
||
@Override | ||
public VoxelShape offset(double x, double y, double z) { | ||
return new VoxelShapeCuboidWithSurfacesInside(this.voxels, this.minX + x, this.minY + y, this.minZ + z, this.maxX + x, this.maxY + y, this.maxZ + z, this.xSegmentSize, this.ySegmentSize, this.zSegmentSize); | ||
} | ||
|
||
@Override | ||
double calculatePenetration(AxisCycleDirection dir, Box box, double maxDist) { | ||
switch (dir) { | ||
case NONE: | ||
return this.calculatePenetration(this.minX, this.maxX, this.xSegmentSize, box.minX, box.maxX, maxDist); | ||
case FORWARD: | ||
return this.calculatePenetration(this.minZ, this.maxZ, this.zSegmentSize, box.minZ, box.maxZ, maxDist); | ||
case BACKWARD: | ||
return this.calculatePenetration(this.minY, this.maxY, this.ySegmentSize, box.minY, box.maxY, maxDist); | ||
default: | ||
throw new IllegalArgumentException(); | ||
} | ||
} | ||
|
||
/** | ||
* Determine how far the movement is possible. | ||
*/ | ||
private double calculatePenetration(double aMin, double aMax, double segmentSize, double bMin, double bMax, double maxDist) { | ||
double gap; | ||
|
||
if (maxDist > 0.0D) { | ||
gap = aMin - bMax; | ||
|
||
if (gap < -EPSILON) { | ||
//already far enough inside this shape | ||
if (segmentSize == Double.POSITIVE_INFINITY) { | ||
//no extra segments to collide with, because only one segment in total | ||
return maxDist; | ||
} | ||
//extra hitboxes inside this shape, evenly spaced out by segmentSize | ||
double distanceToEnd = aMax - bMax; | ||
int segmentsToEnd = (int)(distanceToEnd / segmentSize); | ||
double distanceToSegment = distanceToEnd - ((segmentsToEnd + 1) * segmentSize); | ||
//apply a possible backwards movement because vanilla has it (vanilla: very slightly behind hitbox -> move back) | ||
if (distanceToSegment > -EPSILON) { | ||
segmentsToEnd++; | ||
} | ||
if (segmentsToEnd > 0) | ||
//limit movement to the next segment-segment hitbox | ||
maxDist = Math.min(maxDist, distanceToEnd - segmentsToEnd * segmentSize); | ||
return maxDist; | ||
} else if (maxDist < gap) { | ||
//outside the shape and still far enough away for no collision at all | ||
return maxDist; | ||
|
||
//not really inside the shape and collision is happening | ||
} else if (gap < 0.0D) { //also gap >= -EPSILON here | ||
//already slightly inside the shape but within the margin | ||
//vanilla ignores movement in this case but instead tries to adjust position to be outside the shape again | ||
//cf. these lines in VoxelShape.calculateMaxDistance | ||
// if (g >= -1.0E-7D) { | ||
// maxDist = Math.min(maxDist, g); | ||
//} | ||
return -gap; | ||
} | ||
//allow moving up to the shape but not into it | ||
return gap; | ||
} else { | ||
//whole code again, just negated | ||
gap = aMax - bMin; | ||
|
||
if (gap > EPSILON) { | ||
if (segmentSize == Double.POSITIVE_INFINITY) { | ||
return maxDist; | ||
} | ||
double negDistanceToEnd = aMin - bMin; | ||
int negSegmentsToEnd = (int)(negDistanceToEnd / segmentSize); | ||
double negDistanceToSegment = negDistanceToEnd - ((negSegmentsToEnd - 1) * segmentSize); | ||
if (negDistanceToSegment < EPSILON) { | ||
negSegmentsToEnd--; | ||
} | ||
if (negSegmentsToEnd < 0) | ||
maxDist = Math.max(maxDist, negDistanceToEnd - negSegmentsToEnd * segmentSize); | ||
return maxDist; | ||
} else if (maxDist > gap) { | ||
return maxDist; | ||
} else if (gap > 0.0D) { | ||
return -gap; | ||
} | ||
return gap; | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters