236 lines
11 KiB
Java
236 lines
11 KiB
Java
|
package net.minecraft.world.level.block;
|
||
|
|
||
|
import com.mojang.math.OctahedralGroup;
|
||
|
import com.mojang.math.Quadrant;
|
||
|
import com.mojang.serialization.MapCodec;
|
||
|
import com.mojang.serialization.codecs.RecordCodecBuilder;
|
||
|
import com.mojang.serialization.codecs.RecordCodecBuilder.Instance;
|
||
|
import java.util.Map;
|
||
|
import net.minecraft.core.BlockPos;
|
||
|
import net.minecraft.core.Direction;
|
||
|
import net.minecraft.util.RandomSource;
|
||
|
import net.minecraft.world.item.context.BlockPlaceContext;
|
||
|
import net.minecraft.world.level.BlockGetter;
|
||
|
import net.minecraft.world.level.LevelReader;
|
||
|
import net.minecraft.world.level.ScheduledTickAccess;
|
||
|
import net.minecraft.world.level.block.state.BlockBehaviour;
|
||
|
import net.minecraft.world.level.block.state.BlockState;
|
||
|
import net.minecraft.world.level.block.state.StateDefinition;
|
||
|
import net.minecraft.world.level.block.state.properties.BlockStateProperties;
|
||
|
import net.minecraft.world.level.block.state.properties.BooleanProperty;
|
||
|
import net.minecraft.world.level.block.state.properties.EnumProperty;
|
||
|
import net.minecraft.world.level.block.state.properties.Half;
|
||
|
import net.minecraft.world.level.block.state.properties.StairsShape;
|
||
|
import net.minecraft.world.level.material.FluidState;
|
||
|
import net.minecraft.world.level.material.Fluids;
|
||
|
import net.minecraft.world.level.pathfinder.PathComputationType;
|
||
|
import net.minecraft.world.phys.shapes.CollisionContext;
|
||
|
import net.minecraft.world.phys.shapes.Shapes;
|
||
|
import net.minecraft.world.phys.shapes.VoxelShape;
|
||
|
|
||
|
public class StairBlock extends Block implements SimpleWaterloggedBlock {
|
||
|
public static final MapCodec<StairBlock> CODEC = RecordCodecBuilder.mapCodec(
|
||
|
p_360454_ -> p_360454_.group(BlockState.CODEC.fieldOf("base_state").forGetter(p_309296_ -> p_309296_.baseState), propertiesCodec())
|
||
|
.apply(p_360454_, StairBlock::new)
|
||
|
);
|
||
|
public static final EnumProperty<Direction> FACING = HorizontalDirectionalBlock.FACING;
|
||
|
public static final EnumProperty<Half> HALF = BlockStateProperties.HALF;
|
||
|
public static final EnumProperty<StairsShape> SHAPE = BlockStateProperties.STAIRS_SHAPE;
|
||
|
public static final BooleanProperty WATERLOGGED = BlockStateProperties.WATERLOGGED;
|
||
|
private static final VoxelShape SHAPE_OUTER = Shapes.or(Block.column(16.0, 0.0, 8.0), Block.box(0.0, 8.0, 0.0, 8.0, 16.0, 8.0));
|
||
|
private static final VoxelShape SHAPE_STRAIGHT = Shapes.or(SHAPE_OUTER, Shapes.rotate(SHAPE_OUTER, OctahedralGroup.fromXYAngles(Quadrant.R0, Quadrant.R90)));
|
||
|
private static final VoxelShape SHAPE_INNER = Shapes.or(SHAPE_STRAIGHT, Shapes.rotate(SHAPE_STRAIGHT, OctahedralGroup.fromXYAngles(Quadrant.R0, Quadrant.R90)));
|
||
|
private static final Map<Direction, VoxelShape> SHAPE_BOTTOM_OUTER = Shapes.rotateHorizontal(SHAPE_OUTER);
|
||
|
private static final Map<Direction, VoxelShape> SHAPE_BOTTOM_STRAIGHT = Shapes.rotateHorizontal(SHAPE_STRAIGHT);
|
||
|
private static final Map<Direction, VoxelShape> SHAPE_BOTTOM_INNER = Shapes.rotateHorizontal(SHAPE_INNER);
|
||
|
private static final Map<Direction, VoxelShape> SHAPE_TOP_OUTER = Shapes.rotateHorizontal(Shapes.rotate(SHAPE_OUTER, OctahedralGroup.INVERT_Y));
|
||
|
private static final Map<Direction, VoxelShape> SHAPE_TOP_STRAIGHT = Shapes.rotateHorizontal(Shapes.rotate(SHAPE_STRAIGHT, OctahedralGroup.INVERT_Y));
|
||
|
private static final Map<Direction, VoxelShape> SHAPE_TOP_INNER = Shapes.rotateHorizontal(Shapes.rotate(SHAPE_INNER, OctahedralGroup.INVERT_Y));
|
||
|
private final Block base;
|
||
|
protected final BlockState baseState;
|
||
|
|
||
|
@Override
|
||
|
public MapCodec<? extends StairBlock> codec() {
|
||
|
return CODEC;
|
||
|
}
|
||
|
|
||
|
protected StairBlock(BlockState p_56862_, BlockBehaviour.Properties p_56863_) {
|
||
|
super(p_56863_);
|
||
|
this.registerDefaultState(
|
||
|
this.stateDefinition
|
||
|
.any()
|
||
|
.setValue(FACING, Direction.NORTH)
|
||
|
.setValue(HALF, Half.BOTTOM)
|
||
|
.setValue(SHAPE, StairsShape.STRAIGHT)
|
||
|
.setValue(WATERLOGGED, false)
|
||
|
);
|
||
|
this.base = p_56862_.getBlock();
|
||
|
this.baseState = p_56862_;
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
protected boolean useShapeForLightOcclusion(BlockState p_56967_) {
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
protected VoxelShape getShape(BlockState p_56956_, BlockGetter p_56957_, BlockPos p_56958_, CollisionContext p_56959_) {
|
||
|
boolean flag = p_56956_.getValue(HALF) == Half.BOTTOM;
|
||
|
Direction direction = p_56956_.getValue(FACING);
|
||
|
|
||
|
Map map = switch ((StairsShape)p_56956_.getValue(SHAPE)) {
|
||
|
case STRAIGHT -> flag ? SHAPE_BOTTOM_STRAIGHT : SHAPE_TOP_STRAIGHT;
|
||
|
case OUTER_LEFT, OUTER_RIGHT -> flag ? SHAPE_BOTTOM_OUTER : SHAPE_TOP_OUTER;
|
||
|
case INNER_RIGHT, INNER_LEFT -> flag ? SHAPE_BOTTOM_INNER : SHAPE_TOP_INNER;
|
||
|
};
|
||
|
|
||
|
return (VoxelShape)map.get(switch ((StairsShape)p_56956_.getValue(SHAPE)) {
|
||
|
case STRAIGHT, OUTER_LEFT, INNER_RIGHT -> direction;
|
||
|
case INNER_LEFT -> direction.getCounterClockWise();
|
||
|
case OUTER_RIGHT -> direction.getClockWise();
|
||
|
});
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
public float getExplosionResistance() {
|
||
|
return this.base.getExplosionResistance();
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
public BlockState getStateForPlacement(BlockPlaceContext p_56872_) {
|
||
|
Direction direction = p_56872_.getClickedFace();
|
||
|
BlockPos blockpos = p_56872_.getClickedPos();
|
||
|
FluidState fluidstate = p_56872_.getLevel().getFluidState(blockpos);
|
||
|
BlockState blockstate = this.defaultBlockState()
|
||
|
.setValue(FACING, p_56872_.getHorizontalDirection())
|
||
|
.setValue(
|
||
|
HALF,
|
||
|
direction != Direction.DOWN && (direction == Direction.UP || !(p_56872_.getClickLocation().y - blockpos.getY() > 0.5))
|
||
|
? Half.BOTTOM
|
||
|
: Half.TOP
|
||
|
)
|
||
|
.setValue(WATERLOGGED, fluidstate.getType() == Fluids.WATER);
|
||
|
return blockstate.setValue(SHAPE, getStairsShape(blockstate, p_56872_.getLevel(), blockpos));
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
protected BlockState updateShape(
|
||
|
BlockState p_56925_,
|
||
|
LevelReader p_369543_,
|
||
|
ScheduledTickAccess p_369679_,
|
||
|
BlockPos p_56929_,
|
||
|
Direction p_56926_,
|
||
|
BlockPos p_56930_,
|
||
|
BlockState p_56927_,
|
||
|
RandomSource p_367682_
|
||
|
) {
|
||
|
if (p_56925_.getValue(WATERLOGGED)) {
|
||
|
p_369679_.scheduleTick(p_56929_, Fluids.WATER, Fluids.WATER.getTickDelay(p_369543_));
|
||
|
}
|
||
|
|
||
|
return p_56926_.getAxis().isHorizontal()
|
||
|
? p_56925_.setValue(SHAPE, getStairsShape(p_56925_, p_369543_, p_56929_))
|
||
|
: super.updateShape(p_56925_, p_369543_, p_369679_, p_56929_, p_56926_, p_56930_, p_56927_, p_367682_);
|
||
|
}
|
||
|
|
||
|
private static StairsShape getStairsShape(BlockState p_56977_, BlockGetter p_56978_, BlockPos p_56979_) {
|
||
|
Direction direction = p_56977_.getValue(FACING);
|
||
|
BlockState blockstate = p_56978_.getBlockState(p_56979_.relative(direction));
|
||
|
if (isStairs(blockstate) && p_56977_.getValue(HALF) == blockstate.getValue(HALF)) {
|
||
|
Direction direction1 = blockstate.getValue(FACING);
|
||
|
if (direction1.getAxis() != p_56977_.getValue(FACING).getAxis() && canTakeShape(p_56977_, p_56978_, p_56979_, direction1.getOpposite())) {
|
||
|
if (direction1 == direction.getCounterClockWise()) {
|
||
|
return StairsShape.OUTER_LEFT;
|
||
|
}
|
||
|
|
||
|
return StairsShape.OUTER_RIGHT;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
BlockState blockstate1 = p_56978_.getBlockState(p_56979_.relative(direction.getOpposite()));
|
||
|
if (isStairs(blockstate1) && p_56977_.getValue(HALF) == blockstate1.getValue(HALF)) {
|
||
|
Direction direction2 = blockstate1.getValue(FACING);
|
||
|
if (direction2.getAxis() != p_56977_.getValue(FACING).getAxis() && canTakeShape(p_56977_, p_56978_, p_56979_, direction2)) {
|
||
|
if (direction2 == direction.getCounterClockWise()) {
|
||
|
return StairsShape.INNER_LEFT;
|
||
|
}
|
||
|
|
||
|
return StairsShape.INNER_RIGHT;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return StairsShape.STRAIGHT;
|
||
|
}
|
||
|
|
||
|
private static boolean canTakeShape(BlockState p_56971_, BlockGetter p_56972_, BlockPos p_56973_, Direction p_56974_) {
|
||
|
BlockState blockstate = p_56972_.getBlockState(p_56973_.relative(p_56974_));
|
||
|
return !isStairs(blockstate)
|
||
|
|| blockstate.getValue(FACING) != p_56971_.getValue(FACING)
|
||
|
|| blockstate.getValue(HALF) != p_56971_.getValue(HALF);
|
||
|
}
|
||
|
|
||
|
public static boolean isStairs(BlockState p_56981_) {
|
||
|
return p_56981_.getBlock() instanceof StairBlock;
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
protected BlockState rotate(BlockState p_56922_, Rotation p_56923_) {
|
||
|
return p_56922_.setValue(FACING, p_56923_.rotate(p_56922_.getValue(FACING)));
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
protected BlockState mirror(BlockState p_56919_, Mirror p_56920_) {
|
||
|
Direction direction = p_56919_.getValue(FACING);
|
||
|
StairsShape stairsshape = p_56919_.getValue(SHAPE);
|
||
|
switch (p_56920_) {
|
||
|
case LEFT_RIGHT:
|
||
|
if (direction.getAxis() == Direction.Axis.Z) {
|
||
|
switch (stairsshape) {
|
||
|
case OUTER_LEFT:
|
||
|
return p_56919_.rotate(Rotation.CLOCKWISE_180).setValue(SHAPE, StairsShape.OUTER_RIGHT);
|
||
|
case INNER_RIGHT:
|
||
|
return p_56919_.rotate(Rotation.CLOCKWISE_180).setValue(SHAPE, StairsShape.INNER_LEFT);
|
||
|
case INNER_LEFT:
|
||
|
return p_56919_.rotate(Rotation.CLOCKWISE_180).setValue(SHAPE, StairsShape.INNER_RIGHT);
|
||
|
case OUTER_RIGHT:
|
||
|
return p_56919_.rotate(Rotation.CLOCKWISE_180).setValue(SHAPE, StairsShape.OUTER_LEFT);
|
||
|
default:
|
||
|
return p_56919_.rotate(Rotation.CLOCKWISE_180);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
case FRONT_BACK:
|
||
|
if (direction.getAxis() == Direction.Axis.X) {
|
||
|
switch (stairsshape) {
|
||
|
case STRAIGHT:
|
||
|
return p_56919_.rotate(Rotation.CLOCKWISE_180);
|
||
|
case OUTER_LEFT:
|
||
|
return p_56919_.rotate(Rotation.CLOCKWISE_180).setValue(SHAPE, StairsShape.OUTER_RIGHT);
|
||
|
case INNER_RIGHT:
|
||
|
return p_56919_.rotate(Rotation.CLOCKWISE_180).setValue(SHAPE, StairsShape.INNER_RIGHT);
|
||
|
case INNER_LEFT:
|
||
|
return p_56919_.rotate(Rotation.CLOCKWISE_180).setValue(SHAPE, StairsShape.INNER_LEFT);
|
||
|
case OUTER_RIGHT:
|
||
|
return p_56919_.rotate(Rotation.CLOCKWISE_180).setValue(SHAPE, StairsShape.OUTER_LEFT);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return super.mirror(p_56919_, p_56920_);
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
protected void createBlockStateDefinition(StateDefinition.Builder<Block, BlockState> p_56932_) {
|
||
|
p_56932_.add(FACING, HALF, SHAPE, WATERLOGGED);
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
protected FluidState getFluidState(BlockState p_56969_) {
|
||
|
return p_56969_.getValue(WATERLOGGED) ? Fluids.WATER.getSource(false) : super.getFluidState(p_56969_);
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
protected boolean isPathfindable(BlockState p_56891_, PathComputationType p_56894_) {
|
||
|
return false;
|
||
|
}
|
||
|
}
|