package net.minecraft.world.level.block; import com.mojang.serialization.MapCodec; import net.minecraft.core.BlockPos; import net.minecraft.core.Direction; import net.minecraft.core.particles.ParticleTypes; import net.minecraft.server.level.ServerLevel; import net.minecraft.util.ParticleUtils; import net.minecraft.util.RandomSource; import net.minecraft.util.valueproviders.UniformInt; import net.minecraft.world.item.context.BlockPlaceContext; import net.minecraft.world.level.BlockGetter; import net.minecraft.world.level.Level; 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.levelgen.Heightmap; import net.minecraft.world.level.material.FluidState; import net.minecraft.world.level.material.Fluids; import net.minecraft.world.level.redstone.ExperimentalRedstoneUtils; public class LightningRodBlock extends RodBlock implements SimpleWaterloggedBlock { public static final MapCodec CODEC = simpleCodec(LightningRodBlock::new); public static final BooleanProperty WATERLOGGED = BlockStateProperties.WATERLOGGED; public static final BooleanProperty POWERED = BlockStateProperties.POWERED; private static final int ACTIVATION_TICKS = 8; public static final int RANGE = 128; private static final int SPARK_CYCLE = 200; @Override public MapCodec codec() { return CODEC; } public LightningRodBlock(BlockBehaviour.Properties properties) { super(properties); this.registerDefaultState(this.stateDefinition.any().setValue(FACING, Direction.UP).setValue(WATERLOGGED, false).setValue(POWERED, false)); } @Override public BlockState getStateForPlacement(BlockPlaceContext context) { FluidState fluidState = context.getLevel().getFluidState(context.getClickedPos()); boolean bl = fluidState.getType() == Fluids.WATER; return this.defaultBlockState().setValue(FACING, context.getClickedFace()).setValue(WATERLOGGED, bl); } @Override protected BlockState updateShape( BlockState blockState, LevelReader levelReader, ScheduledTickAccess scheduledTickAccess, BlockPos blockPos, Direction direction, BlockPos blockPos2, BlockState blockState2, RandomSource randomSource ) { if ((Boolean)blockState.getValue(WATERLOGGED)) { scheduledTickAccess.scheduleTick(blockPos, Fluids.WATER, Fluids.WATER.getTickDelay(levelReader)); } return super.updateShape(blockState, levelReader, scheduledTickAccess, blockPos, direction, blockPos2, blockState2, randomSource); } @Override protected FluidState getFluidState(BlockState state) { return state.getValue(WATERLOGGED) ? Fluids.WATER.getSource(false) : super.getFluidState(state); } @Override protected int getSignal(BlockState state, BlockGetter level, BlockPos pos, Direction direction) { return state.getValue(POWERED) ? 15 : 0; } @Override protected int getDirectSignal(BlockState state, BlockGetter level, BlockPos pos, Direction direction) { return state.getValue(POWERED) && state.getValue(FACING) == direction ? 15 : 0; } public void onLightningStrike(BlockState state, Level level, BlockPos pos) { level.setBlock(pos, state.setValue(POWERED, true), 3); this.updateNeighbours(state, level, pos); level.scheduleTick(pos, this, 8); level.levelEvent(3002, pos, ((Direction)state.getValue(FACING)).getAxis().ordinal()); } private void updateNeighbours(BlockState state, Level level, BlockPos pos) { Direction direction = ((Direction)state.getValue(FACING)).getOpposite(); level.updateNeighborsAt(pos.relative(direction), this, ExperimentalRedstoneUtils.initialOrientation(level, direction, null)); } @Override protected void tick(BlockState state, ServerLevel level, BlockPos pos, RandomSource random) { level.setBlock(pos, state.setValue(POWERED, false), 3); this.updateNeighbours(state, level, pos); } @Override public void animateTick(BlockState state, Level level, BlockPos pos, RandomSource random) { if (level.isThundering() && level.random.nextInt(200) <= level.getGameTime() % 200L && pos.getY() == level.getHeight(Heightmap.Types.WORLD_SURFACE, pos.getX(), pos.getZ()) - 1) { ParticleUtils.spawnParticlesAlongAxis(((Direction)state.getValue(FACING)).getAxis(), level, pos, 0.125, ParticleTypes.ELECTRIC_SPARK, UniformInt.of(1, 2)); } } @Override protected void onRemove(BlockState state, Level level, BlockPos pos, BlockState newState, boolean movedByPiston) { if (!state.is(newState.getBlock())) { if ((Boolean)state.getValue(POWERED)) { this.updateNeighbours(state, level, pos); } super.onRemove(state, level, pos, newState, movedByPiston); } } @Override protected void onPlace(BlockState state, Level level, BlockPos pos, BlockState oldState, boolean movedByPiston) { if (!state.is(oldState.getBlock())) { if ((Boolean)state.getValue(POWERED) && !level.getBlockTicks().hasScheduledTick(pos, this)) { level.setBlock(pos, state.setValue(POWERED, false), 18); } } } @Override protected void createBlockStateDefinition(StateDefinition.Builder builder) { builder.add(FACING, POWERED, WATERLOGGED); } @Override protected boolean isSignalSource(BlockState state) { return true; } }