cn.nukkit.math.BlockFace Maven / Gradle / Ivy
package cn.nukkit.math;
import cn.nukkit.api.PowerNukkitOnly;
import cn.nukkit.api.PowerNukkitXOnly;
import cn.nukkit.api.Since;
import com.google.common.collect.Iterators;
import java.util.*;
import java.util.concurrent.ThreadLocalRandom;
import java.util.function.Predicate;
public enum BlockFace {
DOWN(0, 1, -1, "down", AxisDirection.NEGATIVE, new Vector3(0, -1, 0)),
UP(1, 0, -1, "up", AxisDirection.POSITIVE, new Vector3(0, 1, 0)),
NORTH(2, 3, 2, "north", AxisDirection.NEGATIVE, new Vector3(0, 0, -1)),
SOUTH(3, 2, 0, "south", AxisDirection.POSITIVE, new Vector3(0, 0, 1)),
WEST(4, 5, 1, "west", AxisDirection.NEGATIVE, new Vector3(-1, 0, 0)),
EAST(5, 4, 3, "east", AxisDirection.POSITIVE, new Vector3(1, 0, 0));
/**
* All faces in D-U-N-S-W-E order
*/
private static final BlockFace[] VALUES = new BlockFace[6];
/**
* All faces with horizontal axis in order S-W-N-E
*/
private static final BlockFace[] HORIZONTALS = new BlockFace[4];
static {
//Circular dependency
DOWN.axis = Axis.Y;
UP.axis = Axis.Y;
NORTH.axis = Axis.Z;
SOUTH.axis = Axis.Z;
WEST.axis = Axis.X;
EAST.axis = Axis.X;
for (BlockFace face : values()) {
VALUES[face.index] = face;
if (face.getAxis().isHorizontal()) {
HORIZONTALS[face.horizontalIndex] = face;
}
}
}
/**
* Ordering index for D-U-N-S-W-E
*/
private final int index;
/**
* Index of the opposite BlockFace in the VALUES array
*/
private final int opposite;
/**
* Ordering index for the HORIZONTALS field (S-W-N-E)
*/
private final int horizontalIndex;
/**
* The name of this BlockFace (up, down, north, etc.)
*/
private final String name;
private Axis axis;
private final AxisDirection axisDirection;
/**
* Normalized vector that points in the direction of this BlockFace
*/
private final Vector3 unitVector;
BlockFace(int index, int opposite, int horizontalIndex, String name, AxisDirection axisDirection, Vector3 unitVector) {
this.index = index;
this.opposite = opposite;
this.horizontalIndex = horizontalIndex;
this.name = name;
this.axisDirection = axisDirection;
this.unitVector = unitVector;
}
public static BlockFace[] getHorizontals(){
return HORIZONTALS.clone();
}
/**
* Get a BlockFace by it's index (0-5). The order is D-U-N-S-W-E
*
* @param index BlockFace index
* @return block face
*/
public static BlockFace fromIndex(int index) {
return VALUES[MathHelper.abs(index % VALUES.length)];
}
/**
* Get a BlockFace by it's horizontal index (0-3). The order is S-W-N-E
*
* @param index BlockFace index
* @return block face
*/
public static BlockFace fromHorizontalIndex(int index) {
return HORIZONTALS[MathHelper.abs(index % HORIZONTALS.length)];
}
/**
* Get the BlockFace corresponding to the given angle (0-360). An angle of 0 is SOUTH, an angle of 90 would be WEST
*
* @param angle horizontal angle
* @return block face
*/
public static BlockFace fromHorizontalAngle(double angle) {
return fromHorizontalIndex(NukkitMath.floorDouble(angle / 90.0D + 0.5D) & 3);
}
public static BlockFace fromAxis(AxisDirection axisDirection, Axis axis) {
for (BlockFace face : VALUES) {
if (face.getAxisDirection() == axisDirection && face.getAxis() == axis) {
return face;
}
}
throw new RuntimeException("Unable to get face from axis: " + axisDirection + " " + axis);
}
/**
* Choose a random BlockFace using the given Random
*
* @param rand random number generator
* @return block face
*/
public static BlockFace random(Random rand) {
return VALUES[rand.nextInt(VALUES.length)];
}
/**
* Get the index of this BlockFace (0-5). The order is D-U-N-S-W-E
*
* @return index
*/
public int getIndex() {
return index;
}
/**
* Get the index of this BlockFace (0-5). The order is D-U-N-E-S-W
*
* @return index
*/
@PowerNukkitXOnly
@Since("1.6.0.0-PNX")
public int getDUNESWIndex() {
switch (getIndex()) {
case 1:
return 1;
case 2:
return 2;
case 3:
return 4;
case 4:
return 5;
case 5:
return 3;
case 0:
default:
return 0;
}
}
/**
* Get the index of this BlockFace (0-5). The order is D-U-S-W-N-E
*
* @return index
*/
@PowerNukkitXOnly
@Since("1.6.0.0-PNX")
public int getDUSWNEIndex() {
switch (getIndex()) {
case 1:
return 1;
case 2:
return 4;
case 3:
return 2;
case 4:
return 3;
case 5:
return 5;
case 0:
default:
return 0;
}
}
/**
* Get the horizontal index of this BlockFace (0-3). The order is S-W-N-E
*
* @return horizontal index
*/
public int getHorizontalIndex() {
return horizontalIndex;
}
/**
* Get the angle of this BlockFace (0-360)
*
* @return horizontal angle
*/
public float getHorizontalAngle() {
return (float) ((horizontalIndex & 3) * 90);
}
/**
* Get the name of this BlockFace (up, down, north, etc.)
*
* @return name
*/
public String getName() {
return name;
}
/**
* Get the Axis of this BlockFace
*
* @return axis
*/
public Axis getAxis() {
return axis;
}
/**
* Get the AxisDirection of this BlockFace
*
* @return axis direction
*/
public AxisDirection getAxisDirection() {
return axisDirection;
}
/**
* Get the unit vector of this BlockFace
*
* @return vector
*/
public Vector3 getUnitVector() {
return unitVector;
}
/**
* Returns an offset that addresses the block in front of this BlockFace
*
* @return x offset
*/
public int getXOffset() {
return axis == Axis.X ? axisDirection.getOffset() : 0;
}
/**
* Returns an offset that addresses the block in front of this BlockFace
*
* @return y offset
*/
public int getYOffset() {
return axis == Axis.Y ? axisDirection.getOffset() : 0;
}
/**
* Returns an offset that addresses the block in front of this BlockFace
*
* @return x offset
*/
public int getZOffset() {
return axis == Axis.Z ? axisDirection.getOffset() : 0;
}
/**
* Get the opposite BlockFace (e.g. DOWN ==> UP)
*
* @return block face
*/
public BlockFace getOpposite() {
return fromIndex(opposite);
}
/**
* Rotate this BlockFace around the Y axis clockwise (NORTH => EAST => SOUTH => WEST => NORTH)
*
* @return block face
*/
public BlockFace rotateY() {
switch (this) {
case NORTH:
return EAST;
case EAST:
return SOUTH;
case SOUTH:
return WEST;
case WEST:
return NORTH;
default:
throw new RuntimeException("Unable to get Y-rotated face of " + this);
}
}
/**
* Rotate this BlockFace around the Y axis counter-clockwise (NORTH => WEST => SOUTH => EAST => NORTH)
*
* @return block face
*/
public BlockFace rotateYCCW() {
switch (this) {
case NORTH:
return WEST;
case EAST:
return NORTH;
case SOUTH:
return EAST;
case WEST:
return SOUTH;
default:
throw new RuntimeException("Unable to get counter-clockwise Y-rotated face of " + this);
}
}
@PowerNukkitOnly
@Since("1.4.0.0-PN")
public CompassRoseDirection getCompassRoseDirection() {
switch (this) {
case NORTH:
return CompassRoseDirection.NORTH;
case SOUTH:
return CompassRoseDirection.SOUTH;
case WEST:
return CompassRoseDirection.WEST;
case EAST:
return CompassRoseDirection.EAST;
default:
return null;
}
}
@Since("FUTURE")
@PowerNukkitOnly
public Set getEdges() {
EnumSet blockFaces = EnumSet.noneOf(BlockFace.class);
if (axis.isVertical()) {
Collections.addAll(blockFaces, Plane.HORIZONTAL.faces);
return blockFaces;
}
Collections.addAll(blockFaces, Plane.VERTICAL.faces);
Axis edgeAxis = axis == Axis.X ? Axis.Z : Axis.X;
blockFaces.add(fromAxis(AxisDirection.NEGATIVE, edgeAxis));
blockFaces.add(fromAxis(AxisDirection.POSITIVE, edgeAxis));
return blockFaces;
}
@Override
public String toString() {
return name;
}
public enum Axis implements Predicate {
X("x"),
Y("y"),
Z("z");
private final String name;
private Plane plane;
static {
//Circular dependency
X.plane = Plane.HORIZONTAL;
Y.plane = Plane.VERTICAL;
Z.plane = Plane.HORIZONTAL;
}
Axis(String name) {
this.name = name;
}
public boolean isVertical() {
return plane == Plane.VERTICAL;
}
public boolean isHorizontal() {
return plane == Plane.HORIZONTAL;
}
public Plane getPlane() {
return plane;
}
public String getName() {
return name;
}
@Override
public boolean test(BlockFace face) {
return face != null && face.getAxis() == this;
}
@Override
public String toString() {
return name;
}
}
public enum AxisDirection {
POSITIVE(1, "Towards positive"),
NEGATIVE(-1, "Towards negative");
private final int offset;
private final String description;
AxisDirection(int offset, String description) {
this.offset = offset;
this.description = description;
}
public int getOffset() {
return offset;
}
@Override
public String toString() {
return description;
}
}
public enum Plane implements Predicate, Iterable {
HORIZONTAL,
VERTICAL;
static {
//Circular dependency
HORIZONTAL.faces = new BlockFace[]{NORTH, EAST, SOUTH, WEST};
VERTICAL.faces = new BlockFace[]{UP, DOWN};
}
private BlockFace[] faces;
@PowerNukkitOnly
@Since("1.4.0.0-PN")
public BlockFace random() {
return faces[ThreadLocalRandom.current().nextInt(faces.length)];
}
public BlockFace random(NukkitRandom rand) {
return faces[rand.nextBoundedInt(faces.length)];
}
@Override
public boolean test(BlockFace face) {
return face != null && face.getAxis().getPlane() == this;
}
@Override
public Iterator iterator() {
return Iterators.forArray(faces);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy