package net.minecraft.world.level.redstone; import com.google.common.annotations.VisibleForTesting; import io.netty.buffer.ByteBuf; import java.util.EnumMap; import java.util.List; import java.util.Map; import java.util.Objects; import net.minecraft.Util; import net.minecraft.core.Direction; import net.minecraft.core.Vec3i; import net.minecraft.network.codec.ByteBufCodecs; import net.minecraft.network.codec.StreamCodec; import net.minecraft.util.RandomSource; public class Orientation { public static final StreamCodec STREAM_CODEC = ByteBufCodecs.idMapper(Orientation::fromIndex, Orientation::getIndex); private static final Orientation[] ORIENTATIONS = Util.make(() -> { Orientation[] orientations = new Orientation[48]; generateContext(new Orientation(Direction.UP, Direction.NORTH, Orientation.SideBias.LEFT), orientations); return orientations; }); private final Direction up; private final Direction front; private final Direction side; private final Orientation.SideBias sideBias; private final int index; private final List neighbors; private final List horizontalNeighbors; private final List verticalNeighbors; private final Map withFront = new EnumMap(Direction.class); private final Map withUp = new EnumMap(Direction.class); private final Map withSideBias = new EnumMap(Orientation.SideBias.class); private Orientation(Direction up, Direction front, Orientation.SideBias sideBias) { this.up = up; this.front = front; this.sideBias = sideBias; this.index = generateIndex(up, front, sideBias); Vec3i vec3i = front.getUnitVec3i().cross(up.getUnitVec3i()); Direction direction = Direction.getNearest(vec3i, null); Objects.requireNonNull(direction); if (this.sideBias == Orientation.SideBias.RIGHT) { this.side = direction; } else { this.side = direction.getOpposite(); } this.neighbors = List.of(this.front.getOpposite(), this.front, this.side, this.side.getOpposite(), this.up.getOpposite(), this.up); this.horizontalNeighbors = this.neighbors.stream().filter(directionx -> directionx.getAxis() != this.up.getAxis()).toList(); this.verticalNeighbors = this.neighbors.stream().filter(directionx -> directionx.getAxis() == this.up.getAxis()).toList(); } public static Orientation of(Direction up, Direction front, Orientation.SideBias sideBias) { return ORIENTATIONS[generateIndex(up, front, sideBias)]; } public Orientation withUp(Direction up) { return (Orientation)this.withUp.get(up); } public Orientation withFront(Direction front) { return (Orientation)this.withFront.get(front); } public Orientation withFrontPreserveUp(Direction front) { return front.getAxis() == this.up.getAxis() ? this : (Orientation)this.withFront.get(front); } public Orientation withFrontAdjustSideBias(Direction front) { Orientation orientation = this.withFront(front); return this.front == orientation.side ? orientation.withMirror() : orientation; } public Orientation withSideBias(Orientation.SideBias sideBias) { return (Orientation)this.withSideBias.get(sideBias); } public Orientation withMirror() { return this.withSideBias(this.sideBias.getOpposite()); } public Direction getFront() { return this.front; } public Direction getUp() { return this.up; } public Direction getSide() { return this.side; } public Orientation.SideBias getSideBias() { return this.sideBias; } public List getDirections() { return this.neighbors; } public List getHorizontalDirections() { return this.horizontalNeighbors; } public List getVerticalDirections() { return this.verticalNeighbors; } public String toString() { return "[up=" + this.up + ",front=" + this.front + ",sideBias=" + this.sideBias + "]"; } public int getIndex() { return this.index; } public static Orientation fromIndex(int index) { return ORIENTATIONS[index]; } public static Orientation random(RandomSource random) { return Util.getRandom(ORIENTATIONS, random); } private static Orientation generateContext(Orientation start, Orientation[] output) { if (output[start.getIndex()] != null) { return output[start.getIndex()]; } else { output[start.getIndex()] = start; for (Orientation.SideBias sideBias : Orientation.SideBias.values()) { start.withSideBias.put(sideBias, generateContext(new Orientation(start.up, start.front, sideBias), output)); } for (Direction direction : Direction.values()) { Direction direction2 = start.up; if (direction == start.up) { direction2 = start.front.getOpposite(); } if (direction == start.up.getOpposite()) { direction2 = start.front; } start.withFront.put(direction, generateContext(new Orientation(direction2, direction, start.sideBias), output)); } for (Direction direction : Direction.values()) { Direction direction2x = start.front; if (direction == start.front) { direction2x = start.up.getOpposite(); } if (direction == start.front.getOpposite()) { direction2x = start.up; } start.withUp.put(direction, generateContext(new Orientation(direction, direction2x, start.sideBias), output)); } return start; } } @VisibleForTesting protected static int generateIndex(Direction up, Direction front, Orientation.SideBias sideBias) { if (up.getAxis() == front.getAxis()) { throw new IllegalStateException("Up-vector and front-vector can not be on the same axis"); } else { int i; if (up.getAxis() == Direction.Axis.Y) { i = front.getAxis() == Direction.Axis.X ? 1 : 0; } else { i = front.getAxis() == Direction.Axis.Y ? 1 : 0; } int j = i << 1 | front.getAxisDirection().ordinal(); return ((up.ordinal() << 2) + j << 1) + sideBias.ordinal(); } } public static enum SideBias { LEFT("left"), RIGHT("right"); private final String name; private SideBias(final String name) { this.name = name; } public Orientation.SideBias getOpposite() { return this == LEFT ? RIGHT : LEFT; } public String toString() { return this.name; } } }