minecraft-src/net/minecraft/world/level/block/GrowingPlantBodyBlock.java
2025-07-04 01:41:11 +03:00

90 lines
3.6 KiB
Java

package net.minecraft.world.level.block;
import com.mojang.serialization.MapCodec;
import java.util.Optional;
import net.minecraft.BlockUtil;
import net.minecraft.core.BlockPos;
import net.minecraft.core.Direction;
import net.minecraft.server.level.ServerLevel;
import net.minecraft.util.RandomSource;
import net.minecraft.world.item.ItemStack;
import net.minecraft.world.item.context.BlockPlaceContext;
import net.minecraft.world.level.BlockGetter;
import net.minecraft.world.level.Level;
import net.minecraft.world.level.LevelAccessor;
import net.minecraft.world.level.LevelReader;
import net.minecraft.world.level.block.state.BlockBehaviour;
import net.minecraft.world.level.block.state.BlockState;
import net.minecraft.world.level.material.Fluids;
import net.minecraft.world.phys.shapes.VoxelShape;
public abstract class GrowingPlantBodyBlock extends GrowingPlantBlock implements BonemealableBlock {
protected GrowingPlantBodyBlock(BlockBehaviour.Properties properties, Direction growthDirection, VoxelShape shape, boolean scheduleFluidTicks) {
super(properties, growthDirection, shape, scheduleFluidTicks);
}
@Override
protected abstract MapCodec<? extends GrowingPlantBodyBlock> codec();
protected BlockState updateHeadAfterConvertedFromBody(BlockState head, BlockState body) {
return body;
}
@Override
protected BlockState updateShape(BlockState state, Direction direction, BlockState neighborState, LevelAccessor level, BlockPos pos, BlockPos neighborPos) {
if (direction == this.growthDirection.getOpposite() && !state.canSurvive(level, pos)) {
level.scheduleTick(pos, this, 1);
}
GrowingPlantHeadBlock growingPlantHeadBlock = this.getHeadBlock();
if (direction == this.growthDirection && !neighborState.is(this) && !neighborState.is(growingPlantHeadBlock)) {
return this.updateHeadAfterConvertedFromBody(state, growingPlantHeadBlock.getStateForPlacement(level));
} else {
if (this.scheduleFluidTicks) {
level.scheduleTick(pos, Fluids.WATER, Fluids.WATER.getTickDelay(level));
}
return super.updateShape(state, direction, neighborState, level, pos, neighborPos);
}
}
@Override
public ItemStack getCloneItemStack(LevelReader level, BlockPos pos, BlockState state) {
return new ItemStack(this.getHeadBlock());
}
@Override
public boolean isValidBonemealTarget(LevelReader level, BlockPos pos, BlockState state) {
Optional<BlockPos> optional = this.getHeadPos(level, pos, state.getBlock());
return optional.isPresent() && this.getHeadBlock().canGrowInto(level.getBlockState(((BlockPos)optional.get()).relative(this.growthDirection)));
}
@Override
public boolean isBonemealSuccess(Level level, RandomSource random, BlockPos pos, BlockState state) {
return true;
}
@Override
public void performBonemeal(ServerLevel level, RandomSource random, BlockPos pos, BlockState state) {
Optional<BlockPos> optional = this.getHeadPos(level, pos, state.getBlock());
if (optional.isPresent()) {
BlockState blockState = level.getBlockState((BlockPos)optional.get());
((GrowingPlantHeadBlock)blockState.getBlock()).performBonemeal(level, random, (BlockPos)optional.get(), blockState);
}
}
private Optional<BlockPos> getHeadPos(BlockGetter level, BlockPos pos, Block block) {
return BlockUtil.getTopConnectedBlock(level, pos, block, this.growthDirection, this.getHeadBlock());
}
@Override
protected boolean canBeReplaced(BlockState state, BlockPlaceContext useContext) {
boolean bl = super.canBeReplaced(state, useContext);
return bl && useContext.getItemInHand().is(this.getHeadBlock().asItem()) ? false : bl;
}
@Override
protected Block getBodyBlock() {
return this;
}
}