mirror of
https://github.com/Minestom/Minestom.git
synced 2025-01-25 01:21:20 +01:00
More error fix
This commit is contained in:
parent
86f7fa7e27
commit
b15422f406
@ -3,7 +3,6 @@ package net.minestom.server.command.builder.arguments.minecraft.registry;
|
|||||||
import net.minestom.server.command.builder.NodeMaker;
|
import net.minestom.server.command.builder.NodeMaker;
|
||||||
import net.minestom.server.instance.block.Block;
|
import net.minestom.server.instance.block.Block;
|
||||||
import net.minestom.server.network.packet.server.play.DeclareCommandsPacket;
|
import net.minestom.server.network.packet.server.play.DeclareCommandsPacket;
|
||||||
import net.minestom.server.registry.Registries;
|
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
|
||||||
public class ArgumentBlockState extends ArgumentRegistry<Block> {
|
public class ArgumentBlockState extends ArgumentRegistry<Block> {
|
||||||
@ -14,7 +13,7 @@ public class ArgumentBlockState extends ArgumentRegistry<Block> {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Block getRegistry(@NotNull String value) {
|
public Block getRegistry(@NotNull String value) {
|
||||||
return Registries.getBlock(value);
|
return Block.fromNamespaceId(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -4,39 +4,54 @@ import net.minestom.server.MinecraftServer;
|
|||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.event.CancellableEvent;
|
import net.minestom.server.event.CancellableEvent;
|
||||||
import net.minestom.server.event.PlayerEvent;
|
import net.minestom.server.event.PlayerEvent;
|
||||||
|
import net.minestom.server.instance.block.Block;
|
||||||
import net.minestom.server.instance.block.BlockManager;
|
import net.minestom.server.instance.block.BlockManager;
|
||||||
import net.minestom.server.instance.block.CustomBlock;
|
|
||||||
import net.minestom.server.utils.BlockPosition;
|
import net.minestom.server.utils.BlockPosition;
|
||||||
import net.minestom.server.utils.validate.Check;
|
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
import org.jetbrains.annotations.Nullable;
|
|
||||||
|
|
||||||
public class PlayerBlockBreakEvent extends PlayerEvent implements CancellableEvent {
|
public class PlayerBlockBreakEvent extends PlayerEvent implements CancellableEvent {
|
||||||
|
|
||||||
private static final BlockManager BLOCK_MANAGER = MinecraftServer.getBlockManager();
|
private static final BlockManager BLOCK_MANAGER = MinecraftServer.getBlockManager();
|
||||||
|
|
||||||
|
private final Block block;
|
||||||
|
private Block resultBlock;
|
||||||
private final BlockPosition blockPosition;
|
private final BlockPosition blockPosition;
|
||||||
|
|
||||||
private final short blockStateId;
|
|
||||||
private final CustomBlock customBlock;
|
|
||||||
|
|
||||||
private short resultBlockStateId;
|
|
||||||
private short resultCustomBlockId;
|
|
||||||
|
|
||||||
private boolean cancelled;
|
private boolean cancelled;
|
||||||
|
|
||||||
public PlayerBlockBreakEvent(@NotNull Player player, @NotNull BlockPosition blockPosition,
|
public PlayerBlockBreakEvent(@NotNull Player player,
|
||||||
short blockStateId, @Nullable CustomBlock customBlock,
|
@NotNull Block block, @NotNull Block resultBlock, @NotNull BlockPosition blockPosition) {
|
||||||
short resultBlockStateId, short resultCustomBlockId) {
|
|
||||||
super(player);
|
super(player);
|
||||||
|
this.block = block;
|
||||||
|
this.resultBlock = resultBlock;
|
||||||
this.blockPosition = blockPosition;
|
this.blockPosition = blockPosition;
|
||||||
|
}
|
||||||
|
|
||||||
this.blockStateId = blockStateId;
|
/**
|
||||||
this.customBlock = customBlock;
|
* Gets the block to break
|
||||||
|
*
|
||||||
|
* @return the block
|
||||||
|
*/
|
||||||
|
public @NotNull Block getBlock() {
|
||||||
|
return block;
|
||||||
|
}
|
||||||
|
|
||||||
this.resultBlockStateId = resultBlockStateId;
|
/**
|
||||||
this.resultCustomBlockId = resultCustomBlockId;
|
* Gets the block which will replace {@link #getBlock()}.
|
||||||
|
*
|
||||||
|
* @return the result block
|
||||||
|
*/
|
||||||
|
public @NotNull Block getResultBlock() {
|
||||||
|
return resultBlock;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Changes the result of the event.
|
||||||
|
*
|
||||||
|
* @param resultBlock the new block
|
||||||
|
*/
|
||||||
|
public void setResultBlock(@NotNull Block resultBlock) {
|
||||||
|
this.resultBlock = resultBlock;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -44,104 +59,10 @@ public class PlayerBlockBreakEvent extends PlayerEvent implements CancellableEve
|
|||||||
*
|
*
|
||||||
* @return the block position
|
* @return the block position
|
||||||
*/
|
*/
|
||||||
@NotNull
|
public @NotNull BlockPosition getBlockPosition() {
|
||||||
public BlockPosition getBlockPosition() {
|
|
||||||
return blockPosition;
|
return blockPosition;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets the broken block state id.
|
|
||||||
*
|
|
||||||
* @return the block id
|
|
||||||
*/
|
|
||||||
public short getBlockStateId() {
|
|
||||||
return blockStateId;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets the broken custom block.
|
|
||||||
*
|
|
||||||
* @return the custom block,
|
|
||||||
* null if not any
|
|
||||||
*/
|
|
||||||
@Nullable
|
|
||||||
public CustomBlock getCustomBlock() {
|
|
||||||
return customBlock;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets the visual block id result, which will be placed after the event.
|
|
||||||
*
|
|
||||||
* @return the block id that will be set at {@link #getBlockPosition()}
|
|
||||||
* set to 0 to remove
|
|
||||||
*/
|
|
||||||
public short getResultBlockStateId() {
|
|
||||||
return resultBlockStateId;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Changes the visual block id result.
|
|
||||||
*
|
|
||||||
* @param resultBlockStateId the result block id
|
|
||||||
*/
|
|
||||||
public void setResultBlockId(short resultBlockStateId) {
|
|
||||||
this.resultBlockStateId = resultBlockStateId;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets the custom block id result, which will be placed after the event.
|
|
||||||
* <p>
|
|
||||||
* Warning: the visual block will not be changed, be sure to call {@link #setResultBlockId(short)}
|
|
||||||
* if you want the visual to be the same as {@link CustomBlock#getDefaultBlockStateId()}.
|
|
||||||
*
|
|
||||||
* @return the custom block id that will be set at {@link #getBlockPosition()}
|
|
||||||
* set to 0 to remove
|
|
||||||
*/
|
|
||||||
public short getResultCustomBlockId() {
|
|
||||||
return resultCustomBlockId;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Changes the custom block id result, which will be placed after the event.
|
|
||||||
*
|
|
||||||
* @param resultCustomBlockId the custom block id result
|
|
||||||
*/
|
|
||||||
public void setResultCustomBlockId(short resultCustomBlockId) {
|
|
||||||
this.resultCustomBlockId = resultCustomBlockId;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Sets both the blockId and customBlockId.
|
|
||||||
*
|
|
||||||
* @param customBlock the result custom block
|
|
||||||
*/
|
|
||||||
public void setResultCustomBlock(@NotNull CustomBlock customBlock) {
|
|
||||||
setResultBlockId(customBlock.getDefaultBlockStateId());
|
|
||||||
setResultCustomBlockId(customBlock.getCustomBlockId());
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Sets both the blockStateId and customBlockId.
|
|
||||||
*
|
|
||||||
* @param customBlockId the result custom block
|
|
||||||
*/
|
|
||||||
public void setResultCustomBlock(short customBlockId) {
|
|
||||||
final CustomBlock customBlock = BLOCK_MANAGER.getCustomBlock(customBlockId);
|
|
||||||
Check.notNull(customBlock, "The custom block with the id '" + customBlockId + "' does not exist");
|
|
||||||
setResultCustomBlock(customBlock);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Sets both the blockId and customBlockId.
|
|
||||||
*
|
|
||||||
* @param customBlockId the result custom block id
|
|
||||||
*/
|
|
||||||
public void setResultCustomBlock(@NotNull String customBlockId) {
|
|
||||||
final CustomBlock customBlock = BLOCK_MANAGER.getCustomBlock(customBlockId);
|
|
||||||
Check.notNull(customBlock, "The custom block with the identifier '" + customBlockId + "' does not exist");
|
|
||||||
setResultCustomBlock(customBlock);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean isCancelled() {
|
public boolean isCancelled() {
|
||||||
return cancelled;
|
return cancelled;
|
||||||
|
@ -3,6 +3,7 @@ package net.minestom.server.event.player;
|
|||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.event.CancellableEvent;
|
import net.minestom.server.event.CancellableEvent;
|
||||||
import net.minestom.server.event.PlayerEvent;
|
import net.minestom.server.event.PlayerEvent;
|
||||||
|
import net.minestom.server.instance.block.Block;
|
||||||
import net.minestom.server.utils.BlockPosition;
|
import net.minestom.server.utils.BlockPosition;
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
|
||||||
@ -16,17 +17,24 @@ import org.jetbrains.annotations.NotNull;
|
|||||||
*/
|
*/
|
||||||
public class PlayerStartDiggingEvent extends PlayerEvent implements CancellableEvent {
|
public class PlayerStartDiggingEvent extends PlayerEvent implements CancellableEvent {
|
||||||
|
|
||||||
|
private final Block block;
|
||||||
private final BlockPosition blockPosition;
|
private final BlockPosition blockPosition;
|
||||||
private final int blockStateId;
|
|
||||||
private final int customBlockId;
|
|
||||||
|
|
||||||
private boolean cancelled;
|
private boolean cancelled;
|
||||||
|
|
||||||
public PlayerStartDiggingEvent(@NotNull Player player, @NotNull BlockPosition blockPosition, int blockStateId, int customBlockId) {
|
public PlayerStartDiggingEvent(@NotNull Player player, @NotNull Block block, @NotNull BlockPosition blockPosition) {
|
||||||
super(player);
|
super(player);
|
||||||
|
this.block = block;
|
||||||
this.blockPosition = blockPosition;
|
this.blockPosition = blockPosition;
|
||||||
this.blockStateId = blockStateId;
|
}
|
||||||
this.customBlockId = customBlockId;
|
|
||||||
|
/**
|
||||||
|
* Gets the block which is being dug.
|
||||||
|
*
|
||||||
|
* @return the block
|
||||||
|
*/
|
||||||
|
public @NotNull Block getBlock() {
|
||||||
|
return block;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -34,29 +42,10 @@ public class PlayerStartDiggingEvent extends PlayerEvent implements CancellableE
|
|||||||
*
|
*
|
||||||
* @return the {@link BlockPosition}
|
* @return the {@link BlockPosition}
|
||||||
*/
|
*/
|
||||||
@NotNull
|
public @NotNull BlockPosition getBlockPosition() {
|
||||||
public BlockPosition getBlockPosition() {
|
|
||||||
return blockPosition;
|
return blockPosition;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets the block state id.
|
|
||||||
*
|
|
||||||
* @return the block state id
|
|
||||||
*/
|
|
||||||
public int getBlockStateId() {
|
|
||||||
return blockStateId;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets the custom block id.
|
|
||||||
*
|
|
||||||
* @return the custom block id
|
|
||||||
*/
|
|
||||||
public int getCustomBlockId() {
|
|
||||||
return customBlockId;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean isCancelled() {
|
public boolean isCancelled() {
|
||||||
return cancelled;
|
return cancelled;
|
||||||
|
@ -85,7 +85,7 @@ public class DynamicChunk extends Chunk {
|
|||||||
// Update pathfinder
|
// Update pathfinder
|
||||||
if (columnarSpace != null) {
|
if (columnarSpace != null) {
|
||||||
final ColumnarOcclusionFieldList columnarOcclusionFieldList = columnarSpace.occlusionFields();
|
final ColumnarOcclusionFieldList columnarOcclusionFieldList = columnarSpace.occlusionFields();
|
||||||
final PFBlockDescription blockDescription = PFBlockDescription.getBlockDescription(blockStateId);
|
final PFBlockDescription blockDescription = PFBlockDescription.getBlockDescription(block);
|
||||||
columnarOcclusionFieldList.onBlockChanged(x, y, z, blockDescription, 0);
|
columnarOcclusionFieldList.onBlockChanged(x, y, z, blockDescription, 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -18,7 +18,6 @@ import net.minestom.server.storage.StorageLocation;
|
|||||||
import net.minestom.server.utils.BlockPosition;
|
import net.minestom.server.utils.BlockPosition;
|
||||||
import net.minestom.server.utils.PacketUtils;
|
import net.minestom.server.utils.PacketUtils;
|
||||||
import net.minestom.server.utils.Position;
|
import net.minestom.server.utils.Position;
|
||||||
import net.minestom.server.utils.block.CustomBlockUtils;
|
|
||||||
import net.minestom.server.utils.callback.OptionalCallback;
|
import net.minestom.server.utils.callback.OptionalCallback;
|
||||||
import net.minestom.server.utils.chunk.ChunkCallback;
|
import net.minestom.server.utils.chunk.ChunkCallback;
|
||||||
import net.minestom.server.utils.chunk.ChunkSupplier;
|
import net.minestom.server.utils.chunk.ChunkSupplier;
|
||||||
@ -142,21 +141,12 @@ public class InstanceContainer extends Instance {
|
|||||||
|
|
||||||
final BlockPosition blockPosition = new BlockPosition(x, y, z);
|
final BlockPosition blockPosition = new BlockPosition(x, y, z);
|
||||||
|
|
||||||
if (isAlreadyChanged(blockPosition, blockStateId)) { // do NOT change the block again.
|
if (isAlreadyChanged(blockPosition, block)) { // do NOT change the block again.
|
||||||
// Avoids StackOverflowExceptions when onDestroy tries to destroy the block itself
|
// Avoids StackOverflowExceptions when onDestroy tries to destroy the block itself
|
||||||
// This can happen with nether portals which break the entire frame when a portal block is broken
|
// This can happen with nether portals which break the entire frame when a portal block is broken
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
setAlreadyChanged(blockPosition, blockStateId);
|
setAlreadyChanged(blockPosition, block);
|
||||||
|
|
||||||
final int index = ChunkUtils.getBlockIndex(x, y, z);
|
|
||||||
|
|
||||||
final CustomBlock previousBlock = chunk.getCustomBlock(index);
|
|
||||||
final Data previousBlockData = previousBlock != null ? chunk.getBlockData(index) : null;
|
|
||||||
if (previousBlock != null) {
|
|
||||||
// Remove digging information for the previous custom block
|
|
||||||
previousBlock.removeDiggingInformation(this, blockPosition);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Change id based on neighbors
|
// Change id based on neighbors
|
||||||
block = executeBlockPlacementRule(block, blockPosition);
|
block = executeBlockPlacementRule(block, blockPosition);
|
||||||
@ -169,21 +159,11 @@ public class InstanceContainer extends Instance {
|
|||||||
|
|
||||||
// Refresh player chunk block
|
// Refresh player chunk block
|
||||||
sendBlockChange(chunk, blockPosition, block);
|
sendBlockChange(chunk, blockPosition, block);
|
||||||
|
|
||||||
// Call the destroy listener for the previously destroyed block
|
|
||||||
if (previousBlock != null) {
|
|
||||||
callBlockDestroy(previousBlock, previousBlockData, blockPosition);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Call the place listener for newly placed custom block
|
|
||||||
if (isCustomBlock) {
|
|
||||||
callBlockPlace(chunk, index, blockPosition);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void setAlreadyChanged(@NotNull BlockPosition blockPosition, short blockStateId) {
|
private void setAlreadyChanged(@NotNull BlockPosition blockPosition, Block block) {
|
||||||
currentlyChangingBlocks.put(blockPosition, Block.fromStateId(blockStateId));
|
currentlyChangingBlocks.put(blockPosition, block);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -191,14 +171,14 @@ public class InstanceContainer extends Instance {
|
|||||||
* Prevents StackOverflow with blocks trying to modify their position in onDestroy or onPlace.
|
* Prevents StackOverflow with blocks trying to modify their position in onDestroy or onPlace.
|
||||||
*
|
*
|
||||||
* @param blockPosition the block position
|
* @param blockPosition the block position
|
||||||
* @param blockStateId the block state id
|
* @param block the block
|
||||||
* @return true if the block changed since the last update
|
* @return true if the block changed since the last update
|
||||||
*/
|
*/
|
||||||
private boolean isAlreadyChanged(@NotNull BlockPosition blockPosition, short blockStateId) {
|
private boolean isAlreadyChanged(@NotNull BlockPosition blockPosition, @NotNull Block block) {
|
||||||
final Block changedBlock = currentlyChangingBlocks.get(blockPosition);
|
final Block changedBlock = currentlyChangingBlocks.get(blockPosition);
|
||||||
if (changedBlock == null)
|
if (changedBlock == null)
|
||||||
return false;
|
return false;
|
||||||
return changedBlock.getBlockId() == blockStateId;
|
return changedBlock.getBlockId() == block.getBlockId();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -215,23 +195,6 @@ public class InstanceContainer extends Instance {
|
|||||||
previousBlock.onDestroy(this, blockPosition, previousBlockData);
|
previousBlock.onDestroy(this, blockPosition, previousBlockData);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Calls {@link CustomBlock#onPlace(Instance, BlockPosition, Data)} for the current custom block at the position.
|
|
||||||
* <p>
|
|
||||||
* WARNING {@code chunk} needs to be synchronized.
|
|
||||||
*
|
|
||||||
* @param chunk the chunk where the block is
|
|
||||||
* @param index the block index
|
|
||||||
* @param blockPosition the block position
|
|
||||||
*/
|
|
||||||
private void callBlockPlace(@NotNull Chunk chunk, int index, @NotNull BlockPosition blockPosition) {
|
|
||||||
final CustomBlock actualBlock = chunk.getCustomBlock(index);
|
|
||||||
if (actualBlock == null)
|
|
||||||
return;
|
|
||||||
final Data previousData = chunk.getBlockData(index);
|
|
||||||
actualBlock.onPlace(this, blockPosition, previousData);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Calls the {@link BlockPlacementRule} for the specified block state id.
|
* Calls the {@link BlockPlacementRule} for the specified block state id.
|
||||||
*
|
*
|
||||||
@ -244,7 +207,7 @@ public class InstanceContainer extends Instance {
|
|||||||
if (blockPlacementRule != null) {
|
if (blockPlacementRule != null) {
|
||||||
return blockPlacementRule.blockUpdate(this, blockPosition, block);
|
return blockPlacementRule.blockUpdate(this, blockPosition, block);
|
||||||
}
|
}
|
||||||
return blockStateId;
|
return block;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -269,28 +232,16 @@ public class InstanceContainer extends Instance {
|
|||||||
if (chunk == null)
|
if (chunk == null)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
final short neighborStateId = chunk.getBlockStateId(neighborX, neighborY, neighborZ);
|
final Block neighborBlock = chunk.getBlock(neighborX, neighborY, neighborZ);
|
||||||
final BlockPlacementRule neighborBlockPlacementRule = BLOCK_MANAGER.getBlockPlacementRule(neighborStateId);
|
final BlockPlacementRule neighborBlockPlacementRule = BLOCK_MANAGER.getBlockPlacementRule(neighborBlock);
|
||||||
if (neighborBlockPlacementRule != null) {
|
if (neighborBlockPlacementRule != null) {
|
||||||
final BlockPosition neighborPosition = new BlockPosition(neighborX, neighborY, neighborZ);
|
final BlockPosition neighborPosition = new BlockPosition(neighborX, neighborY, neighborZ);
|
||||||
final short newNeighborId = neighborBlockPlacementRule.blockUpdate(this,
|
final Block newNeighborBlock = neighborBlockPlacementRule.blockUpdate(this,
|
||||||
neighborPosition, neighborStateId);
|
neighborPosition, neighborBlock);
|
||||||
if (neighborStateId != newNeighborId) {
|
if (neighborBlock != newNeighborBlock) {
|
||||||
refreshBlockStateId(neighborPosition, newNeighborId);
|
setBlock(neighborPosition, newNeighborBlock);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update neighbors
|
|
||||||
final CustomBlock customBlock = getCustomBlock(neighborX, neighborY, neighborZ);
|
|
||||||
if (customBlock != null) {
|
|
||||||
boolean directNeighbor = false; // only if directly connected to neighbor (no diagonals)
|
|
||||||
if (offsetX != 0 ^ offsetZ != 0) {
|
|
||||||
directNeighbor = offsetY == 0;
|
|
||||||
} else if (offsetX == 0 && offsetZ == 0) {
|
|
||||||
directNeighbor = true;
|
|
||||||
}
|
|
||||||
customBlock.updateFromNeighbor(this, new BlockPosition(neighborX, neighborY, neighborZ), blockPosition, directNeighbor);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -316,37 +267,34 @@ public class InstanceContainer extends Instance {
|
|||||||
final int y = blockPosition.getY();
|
final int y = blockPosition.getY();
|
||||||
final int z = blockPosition.getZ();
|
final int z = blockPosition.getZ();
|
||||||
|
|
||||||
final short blockStateId = getBlockStateId(x, y, z);
|
final Block block = getBlock(x, y, z);
|
||||||
|
|
||||||
// The player probably have a wrong version of this chunk section, send it
|
// The player probably have a wrong version of this chunk section, send it
|
||||||
if (blockStateId == 0) {
|
if (block.isAir()) {
|
||||||
chunk.sendChunkSectionUpdate(ChunkUtils.getSectionAt(y), player);
|
chunk.sendChunkSectionUpdate(ChunkUtils.getSectionAt(y), player);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
final CustomBlock customBlock = getCustomBlock(x, y, z);
|
PlayerBlockBreakEvent blockBreakEvent = new PlayerBlockBreakEvent(player, block, Block.AIR, blockPosition);
|
||||||
|
|
||||||
PlayerBlockBreakEvent blockBreakEvent = new PlayerBlockBreakEvent(player, blockPosition, blockStateId, customBlock, (short) 0, (short) 0);
|
|
||||||
player.callEvent(PlayerBlockBreakEvent.class, blockBreakEvent);
|
player.callEvent(PlayerBlockBreakEvent.class, blockBreakEvent);
|
||||||
final boolean allowed = !blockBreakEvent.isCancelled();
|
final boolean allowed = !blockBreakEvent.isCancelled();
|
||||||
if (allowed) {
|
if (allowed) {
|
||||||
// Break or change the broken block based on event result
|
// Break or change the broken block based on event result
|
||||||
final short resultState = blockBreakEvent.getResultBlockStateId();
|
final Block resultBlock = blockBreakEvent.getResultBlock();
|
||||||
final short resultCustom = blockBreakEvent.getResultCustomBlockId();
|
setBlock(x, y, z, resultBlock);
|
||||||
setSeparateBlocks(x, y, z, resultState, resultCustom);
|
|
||||||
|
|
||||||
// Send the block break effect packet
|
// Send the block break effect packet
|
||||||
{
|
{
|
||||||
EffectPacket effectPacket = new EffectPacket();
|
EffectPacket effectPacket = new EffectPacket();
|
||||||
effectPacket.effectId = 2001; // Block break + block break sound
|
effectPacket.effectId = 2001; // Block break + block break sound
|
||||||
effectPacket.position = blockPosition;
|
effectPacket.position = blockPosition;
|
||||||
effectPacket.data = blockStateId;
|
effectPacket.data = resultBlock.getStateId();
|
||||||
effectPacket.disableRelativeVolume = false;
|
effectPacket.disableRelativeVolume = false;
|
||||||
|
|
||||||
PacketUtils.sendGroupedPacket(chunk.getViewers(), effectPacket,
|
PacketUtils.sendGroupedPacket(chunk.getViewers(), effectPacket,
|
||||||
(viewer) -> {
|
(viewer) -> {
|
||||||
// Prevent the block breaker to play the particles and sound two times
|
// Prevent the block breaker to play the particles and sound two times
|
||||||
return (customBlock != null && customBlock.enableCustomBreakDelay()) || !viewer.equals(player);
|
return !viewer.equals(player);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -99,10 +99,7 @@ public class RedstonePlacementRule extends BlockPlacementRule {
|
|||||||
public Block blockPlace(@NotNull Instance instance,
|
public Block blockPlace(@NotNull Instance instance,
|
||||||
@NotNull Block block, @NotNull BlockFace blockFace, @NotNull BlockPosition blockPosition,
|
@NotNull Block block, @NotNull BlockFace blockFace, @NotNull BlockPosition blockPosition,
|
||||||
@NotNull Player pl) {
|
@NotNull Player pl) {
|
||||||
final short belowBlockId = instance.getBlockStateId(blockPosition.getX(), blockPosition.getY() - 1, blockPosition.getZ());
|
final Block belowBlock = instance.getBlock(blockPosition.getX(), blockPosition.getY() - 1, blockPosition.getZ());
|
||||||
if (!Block.fromStateId(belowBlockId).getData().isSolid()) {
|
return belowBlock.isSolid() ? block : null;
|
||||||
return null;
|
|
||||||
}
|
|
||||||
return block;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
package net.minestom.server.listener;
|
package net.minestom.server.listener;
|
||||||
|
|
||||||
import net.minestom.server.MinecraftServer;
|
import net.minestom.server.MinecraftServer;
|
||||||
import net.minestom.server.data.Data;
|
|
||||||
import net.minestom.server.entity.Entity;
|
import net.minestom.server.entity.Entity;
|
||||||
import net.minestom.server.entity.EntityType;
|
import net.minestom.server.entity.EntityType;
|
||||||
import net.minestom.server.entity.GameMode;
|
import net.minestom.server.entity.GameMode;
|
||||||
@ -14,7 +13,6 @@ import net.minestom.server.instance.Instance;
|
|||||||
import net.minestom.server.instance.block.Block;
|
import net.minestom.server.instance.block.Block;
|
||||||
import net.minestom.server.instance.block.BlockFace;
|
import net.minestom.server.instance.block.BlockFace;
|
||||||
import net.minestom.server.instance.block.BlockManager;
|
import net.minestom.server.instance.block.BlockManager;
|
||||||
import net.minestom.server.instance.block.CustomBlock;
|
|
||||||
import net.minestom.server.instance.block.rule.BlockPlacementRule;
|
import net.minestom.server.instance.block.rule.BlockPlacementRule;
|
||||||
import net.minestom.server.inventory.PlayerInventory;
|
import net.minestom.server.inventory.PlayerInventory;
|
||||||
import net.minestom.server.item.ItemStack;
|
import net.minestom.server.item.ItemStack;
|
||||||
@ -53,21 +51,8 @@ public class BlockPlacementListener {
|
|||||||
|
|
||||||
// Interact at block
|
// Interact at block
|
||||||
// FIXME: onUseOnBlock
|
// FIXME: onUseOnBlock
|
||||||
final boolean cancel = false;//usedItem.onUseOnBlock(player, hand, blockPosition, direction);
|
|
||||||
PlayerBlockInteractEvent playerBlockInteractEvent = new PlayerBlockInteractEvent(player, blockPosition, hand, blockFace);
|
PlayerBlockInteractEvent playerBlockInteractEvent = new PlayerBlockInteractEvent(player, blockPosition, hand, blockFace);
|
||||||
playerBlockInteractEvent.setCancelled(cancel);
|
player.callEvent(PlayerBlockInteractEvent.class, playerBlockInteractEvent);
|
||||||
playerBlockInteractEvent.setBlockingItemUse(cancel);
|
|
||||||
player.callCancellableEvent(PlayerBlockInteractEvent.class, playerBlockInteractEvent, () -> {
|
|
||||||
final CustomBlock customBlock = instance.getCustomBlock(blockPosition);
|
|
||||||
if (customBlock != null) {
|
|
||||||
final Data data = instance.getBlockData(blockPosition);
|
|
||||||
final boolean blocksItem = customBlock.onInteract(player, hand, blockPosition, data);
|
|
||||||
if (blocksItem) {
|
|
||||||
playerBlockInteractEvent.setBlockingItemUse(true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
if (playerBlockInteractEvent.isBlockingItemUse()) {
|
if (playerBlockInteractEvent.isBlockingItemUse()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -150,23 +135,16 @@ public class BlockPlacementListener {
|
|||||||
if (!playerBlockPlaceEvent.isCancelled()) {
|
if (!playerBlockPlaceEvent.isCancelled()) {
|
||||||
|
|
||||||
// BlockPlacementRule check
|
// BlockPlacementRule check
|
||||||
short blockStateId = playerBlockPlaceEvent.getBlockStateId();
|
Block resultBlock = playerBlockPlaceEvent.getBlock();
|
||||||
final Block resultBlock = Block.fromStateId(blockStateId);
|
|
||||||
final BlockPlacementRule blockPlacementRule = BLOCK_MANAGER.getBlockPlacementRule(resultBlock);
|
final BlockPlacementRule blockPlacementRule = BLOCK_MANAGER.getBlockPlacementRule(resultBlock);
|
||||||
if (blockPlacementRule != null) {
|
if (blockPlacementRule != null) {
|
||||||
// Get id from block placement rule instead of the event
|
// Get id from block placement rule instead of the event
|
||||||
blockStateId = blockPlacementRule.blockPlace(instance, resultBlock, blockFace, blockPosition, player);
|
resultBlock = blockPlacementRule.blockPlace(instance, resultBlock, blockFace, blockPosition, player);
|
||||||
}
|
}
|
||||||
final boolean placementRuleCheck = blockStateId != BlockPlacementRule.CANCEL_CODE;
|
final boolean placementRuleCheck = resultBlock != null;
|
||||||
|
|
||||||
if (placementRuleCheck) {
|
if (placementRuleCheck) {
|
||||||
|
|
||||||
// Place the block
|
// Place the block
|
||||||
final short customBlockId = playerBlockPlaceEvent.getCustomBlockId();
|
instance.setBlock(blockPosition, resultBlock);
|
||||||
final Data blockData = playerBlockPlaceEvent.getBlockData(); // Possibly null
|
|
||||||
instance.setSeparateBlocks(blockPosition.getX(), blockPosition.getY(), blockPosition.getZ(),
|
|
||||||
blockStateId, customBlockId, blockData);
|
|
||||||
|
|
||||||
// Block consuming
|
// Block consuming
|
||||||
if (playerBlockPlaceEvent.doesConsumeBlock()) {
|
if (playerBlockPlaceEvent.doesConsumeBlock()) {
|
||||||
// Consume the block in the player's hand
|
// Consume the block in the player's hand
|
||||||
|
@ -37,11 +37,11 @@ public class PlayerDiggingListener {
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
if (status == ClientPlayerDiggingPacket.Status.STARTED_DIGGING) {
|
if (status == ClientPlayerDiggingPacket.Status.STARTED_DIGGING) {
|
||||||
final short blockStateId = instance.getBlockStateId(blockPosition);
|
final Block block = instance.getBlock(blockPosition);
|
||||||
|
|
||||||
//Check if the player is allowed to break blocks based on their game mode
|
//Check if the player is allowed to break blocks based on their game mode
|
||||||
if (player.getGameMode() == GameMode.SPECTATOR) {
|
if (player.getGameMode() == GameMode.SPECTATOR) {
|
||||||
sendAcknowledgePacket(player, blockPosition, blockStateId,
|
sendAcknowledgePacket(player, blockPosition, block,
|
||||||
ClientPlayerDiggingPacket.Status.STARTED_DIGGING, false);
|
ClientPlayerDiggingPacket.Status.STARTED_DIGGING, false);
|
||||||
return; //Spectators can't break blocks
|
return; //Spectators can't break blocks
|
||||||
} else if (player.getGameMode() == GameMode.ADVENTURE) {
|
} else if (player.getGameMode() == GameMode.ADVENTURE) {
|
||||||
@ -49,7 +49,7 @@ public class PlayerDiggingListener {
|
|||||||
ItemStack itemInMainHand = player.getItemInMainHand();
|
ItemStack itemInMainHand = player.getItemInMainHand();
|
||||||
Block destroyedBlock = instance.getBlock(blockPosition);
|
Block destroyedBlock = instance.getBlock(blockPosition);
|
||||||
if (!itemInMainHand.getMeta().getCanDestroy().contains(destroyedBlock)) {
|
if (!itemInMainHand.getMeta().getCanDestroy().contains(destroyedBlock)) {
|
||||||
sendAcknowledgePacket(player, blockPosition, blockStateId,
|
sendAcknowledgePacket(player, blockPosition, block,
|
||||||
ClientPlayerDiggingPacket.Status.STARTED_DIGGING, false);
|
ClientPlayerDiggingPacket.Status.STARTED_DIGGING, false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -57,23 +57,21 @@ public class PlayerDiggingListener {
|
|||||||
|
|
||||||
final boolean instantBreak = player.isCreative() ||
|
final boolean instantBreak = player.isCreative() ||
|
||||||
player.isInstantBreak() ||
|
player.isInstantBreak() ||
|
||||||
Block.fromStateId(blockStateId).breaksInstantaneously();
|
block.breaksInstantaneously();
|
||||||
|
|
||||||
if (instantBreak) {
|
if (instantBreak) {
|
||||||
// No need to check custom block
|
// No need to check custom block
|
||||||
breakBlock(instance, player, blockPosition, blockStateId, status);
|
breakBlock(instance, player, blockPosition, block, status);
|
||||||
} else {
|
} else {
|
||||||
final CustomBlock customBlock = instance.getCustomBlock(blockPosition);
|
final CustomBlock customBlock = instance.getCustomBlock(blockPosition);
|
||||||
final int customBlockId = customBlock == null ? 0 : customBlock.getCustomBlockId();
|
PlayerStartDiggingEvent playerStartDiggingEvent = new PlayerStartDiggingEvent(player, block, blockPosition);
|
||||||
|
|
||||||
PlayerStartDiggingEvent playerStartDiggingEvent = new PlayerStartDiggingEvent(player, blockPosition, blockStateId, customBlockId);
|
|
||||||
player.callEvent(PlayerStartDiggingEvent.class, playerStartDiggingEvent);
|
player.callEvent(PlayerStartDiggingEvent.class, playerStartDiggingEvent);
|
||||||
|
|
||||||
if (playerStartDiggingEvent.isCancelled()) {
|
if (playerStartDiggingEvent.isCancelled()) {
|
||||||
addEffect(player);
|
addEffect(player);
|
||||||
|
|
||||||
// Unsuccessful digging
|
// Unsuccessful digging
|
||||||
sendAcknowledgePacket(player, blockPosition, blockStateId,
|
sendAcknowledgePacket(player, blockPosition, block,
|
||||||
ClientPlayerDiggingPacket.Status.STARTED_DIGGING, false);
|
ClientPlayerDiggingPacket.Status.STARTED_DIGGING, false);
|
||||||
} else if (customBlock != null) {
|
} else if (customBlock != null) {
|
||||||
// Start digging the custom block
|
// Start digging the custom block
|
||||||
@ -82,31 +80,31 @@ public class PlayerDiggingListener {
|
|||||||
addEffect(player);
|
addEffect(player);
|
||||||
}
|
}
|
||||||
|
|
||||||
sendAcknowledgePacket(player, blockPosition, blockStateId,
|
sendAcknowledgePacket(player, blockPosition, block,
|
||||||
ClientPlayerDiggingPacket.Status.STARTED_DIGGING, true);
|
ClientPlayerDiggingPacket.Status.STARTED_DIGGING, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
} else if (status == ClientPlayerDiggingPacket.Status.CANCELLED_DIGGING) {
|
} else if (status == ClientPlayerDiggingPacket.Status.CANCELLED_DIGGING) {
|
||||||
|
|
||||||
final short blockStateId = instance.getBlockStateId(blockPosition);
|
final Block block = instance.getBlock(blockPosition);
|
||||||
// Remove custom block target
|
// Remove custom block target
|
||||||
player.resetTargetBlock();
|
player.resetTargetBlock();
|
||||||
|
|
||||||
sendAcknowledgePacket(player, blockPosition, blockStateId,
|
sendAcknowledgePacket(player, blockPosition, block,
|
||||||
ClientPlayerDiggingPacket.Status.CANCELLED_DIGGING, true);
|
ClientPlayerDiggingPacket.Status.CANCELLED_DIGGING, true);
|
||||||
|
|
||||||
} else if (status == ClientPlayerDiggingPacket.Status.FINISHED_DIGGING) {
|
} else if (status == ClientPlayerDiggingPacket.Status.FINISHED_DIGGING) {
|
||||||
|
|
||||||
final short blockStateId = instance.getBlockStateId(blockPosition);
|
final Block block = instance.getBlock(blockPosition);
|
||||||
final CustomBlock customBlock = instance.getCustomBlock(blockPosition);
|
final CustomBlock customBlock = instance.getCustomBlock(blockPosition);
|
||||||
if (customBlock != null && customBlock.enableCustomBreakDelay()) {
|
if (customBlock != null && customBlock.enableCustomBreakDelay()) {
|
||||||
// Is not supposed to happen, probably a bug
|
// Is not supposed to happen, probably a bug
|
||||||
sendAcknowledgePacket(player, blockPosition, blockStateId,
|
sendAcknowledgePacket(player, blockPosition, block,
|
||||||
ClientPlayerDiggingPacket.Status.FINISHED_DIGGING, false);
|
ClientPlayerDiggingPacket.Status.FINISHED_DIGGING, false);
|
||||||
} else {
|
} else {
|
||||||
// Vanilla block
|
// Vanilla block
|
||||||
breakBlock(instance, player, blockPosition, blockStateId, status);
|
breakBlock(instance, player, blockPosition, block, status);
|
||||||
}
|
}
|
||||||
|
|
||||||
} else if (status == ClientPlayerDiggingPacket.Status.DROP_ITEM_STACK) {
|
} else if (status == ClientPlayerDiggingPacket.Status.DROP_ITEM_STACK) {
|
||||||
@ -169,7 +167,7 @@ public class PlayerDiggingListener {
|
|||||||
|
|
||||||
private static void breakBlock(Instance instance,
|
private static void breakBlock(Instance instance,
|
||||||
Player player,
|
Player player,
|
||||||
BlockPosition blockPosition, int blockStateId,
|
BlockPosition blockPosition, Block block,
|
||||||
ClientPlayerDiggingPacket.Status status) {
|
ClientPlayerDiggingPacket.Status status) {
|
||||||
// Finished digging, remove effect if any
|
// Finished digging, remove effect if any
|
||||||
player.resetTargetBlock();
|
player.resetTargetBlock();
|
||||||
@ -177,15 +175,13 @@ public class PlayerDiggingListener {
|
|||||||
// Unverified block break, client is fully responsible
|
// Unverified block break, client is fully responsible
|
||||||
final boolean result = instance.breakBlock(player, blockPosition);
|
final boolean result = instance.breakBlock(player, blockPosition);
|
||||||
|
|
||||||
final int updatedBlockId = instance.getBlockStateId(blockPosition);
|
final Block updatedBlock = instance.getBlock(blockPosition);
|
||||||
|
|
||||||
// Send acknowledge packet to allow or cancel the digging process
|
// Send acknowledge packet to allow or cancel the digging process
|
||||||
sendAcknowledgePacket(player, blockPosition, updatedBlockId,
|
sendAcknowledgePacket(player, blockPosition, updatedBlock, status, result);
|
||||||
status, result);
|
|
||||||
|
|
||||||
if (!result) {
|
if (!result) {
|
||||||
final boolean solid = Block.fromStateId((short) blockStateId).isSolid();
|
if (block.isSolid()) {
|
||||||
if (solid) {
|
|
||||||
final BlockPosition playerBlockPosition = player.getPosition().toBlockPosition();
|
final BlockPosition playerBlockPosition = player.getPosition().toBlockPosition();
|
||||||
|
|
||||||
// Teleport the player back if he broke a solid block just below him
|
// Teleport the player back if he broke a solid block just below him
|
||||||
@ -252,15 +248,15 @@ public class PlayerDiggingListener {
|
|||||||
*
|
*
|
||||||
* @param player the player
|
* @param player the player
|
||||||
* @param blockPosition the block position
|
* @param blockPosition the block position
|
||||||
* @param blockStateId the block state id
|
* @param block the block
|
||||||
* @param status the status of the digging
|
* @param status the status of the digging
|
||||||
* @param success true to notify of a success, false otherwise
|
* @param success true to notify of a success, false otherwise
|
||||||
*/
|
*/
|
||||||
private static void sendAcknowledgePacket(@NotNull Player player, @NotNull BlockPosition blockPosition, int blockStateId,
|
private static void sendAcknowledgePacket(@NotNull Player player, @NotNull BlockPosition blockPosition, Block block,
|
||||||
@NotNull ClientPlayerDiggingPacket.Status status, boolean success) {
|
@NotNull ClientPlayerDiggingPacket.Status status, boolean success) {
|
||||||
AcknowledgePlayerDiggingPacket acknowledgePlayerDiggingPacket = new AcknowledgePlayerDiggingPacket();
|
AcknowledgePlayerDiggingPacket acknowledgePlayerDiggingPacket = new AcknowledgePlayerDiggingPacket();
|
||||||
acknowledgePlayerDiggingPacket.blockPosition = blockPosition;
|
acknowledgePlayerDiggingPacket.blockPosition = blockPosition;
|
||||||
acknowledgePlayerDiggingPacket.blockStateId = blockStateId;
|
acknowledgePlayerDiggingPacket.blockStateId = block.getStateId();
|
||||||
acknowledgePlayerDiggingPacket.status = status;
|
acknowledgePlayerDiggingPacket.status = status;
|
||||||
acknowledgePlayerDiggingPacket.successful = success;
|
acknowledgePlayerDiggingPacket.successful = success;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user