Improves collision detection

This change majorly improves the hit-detection for blocks hit by a player while dropping.
This commit is contained in:
Kristian Knarvik 2023-03-25 12:14:25 +01:00
parent fba75d2c3f
commit 14572de102
2 changed files with 37 additions and 17 deletions

View File

@ -12,6 +12,9 @@ import org.bukkit.event.Listener;
import org.bukkit.event.player.PlayerMoveEvent; import org.bukkit.event.player.PlayerMoveEvent;
import org.bukkit.util.Vector; import org.bukkit.util.Vector;
import java.util.HashSet;
import java.util.Set;
/** /**
* A listener for players moving inside a dropper arena * A listener for players moving inside a dropper arena
*/ */
@ -20,7 +23,7 @@ public class MoveListener implements Listener {
@EventHandler @EventHandler
public void onPlayerMove(PlayerMoveEvent event) { public void onPlayerMove(PlayerMoveEvent event) {
// Ignore if no actual movement is happening // Ignore if no actual movement is happening
if (event.getFrom().equals(event.getTo())) { if (event.getFrom().equals(event.getTo()) || event.getTo() == null) {
return; return;
} }
@ -39,24 +42,21 @@ public class MoveListener implements Listener {
// Only do block type checking if the block beneath the player changes // Only do block type checking if the block beneath the player changes
if (event.getFrom().getBlock() != event.getTo().getBlock()) { if (event.getFrom().getBlock() != event.getTo().getBlock()) {
Block targetBlock = event.getTo().getBlock(); // Check if the player enters water
Material targetBlockType = targetBlock.getType(); for (Block block : getBlocksBeneathLocation(event.getTo(), 0)) {
if (block.getType() == Material.WATER) {
// Hitting water is the trigger for winning arenaSession.triggerWin();
if (targetBlockType == Material.WATER) { return;
arenaSession.triggerWin(); }
return;
} }
Location targetLocation = targetBlock.getLocation(); // Check if the player is about to hit a non-air and non-liquid block
Material beneathPlayerType = targetLocation.getWorld().getBlockAt( for (Block block : getBlocksBeneathLocation(event.getTo(), 1)) {
targetLocation.add(0, -0.1, 0)).getType(); if (!block.getType().isAir() && block.getType() != Material.STRUCTURE_VOID &&
block.getType() != Material.WATER && block.getType() != Material.LAVA) {
// If hitting something which is not air or water, it must be a solid block, and would end in a loss arenaSession.triggerLoss();
if (!targetBlockType.isAir() || (beneathPlayerType != Material.WATER && return;
!beneathPlayerType.isAir())) { }
arenaSession.triggerLoss();
return;
} }
} }
@ -64,6 +64,22 @@ public class MoveListener implements Listener {
updatePlayerVelocity(arenaSession); updatePlayerVelocity(arenaSession);
} }
/**
* Gets the blocks at the given location that will be affected by the player's hit-box
*
* @param location <p>The location to check</p>
* @return <p>The blocks beneath the player</p>
*/
private Set<Block> getBlocksBeneathLocation(Location location, double depth) {
Set<Block> blocksBeneath = new HashSet<>();
double halfPlayerWidth = 0.3;
blocksBeneath.add(location.clone().subtract(halfPlayerWidth, depth, halfPlayerWidth).getBlock());
blocksBeneath.add(location.clone().subtract(-halfPlayerWidth, depth, halfPlayerWidth).getBlock());
blocksBeneath.add(location.clone().subtract(halfPlayerWidth, depth, -halfPlayerWidth).getBlock());
blocksBeneath.add(location.clone().subtract(-halfPlayerWidth, depth, -halfPlayerWidth).getBlock());
return blocksBeneath;
}
/** /**
* Updates the velocity of the player in the given session * Updates the velocity of the player in the given session
* *

View File

@ -22,6 +22,10 @@ public class PlayerLeaveListener implements Listener {
@EventHandler @EventHandler
public void onPlayerTeleport(PlayerTeleportEvent event) { public void onPlayerTeleport(PlayerTeleportEvent event) {
if (event.getTo() == null) {
return;
}
DropperArenaSession arenaSession = getSession(event.getPlayer()); DropperArenaSession arenaSession = getSession(event.getPlayer());
if (arenaSession == null) { if (arenaSession == null) {
return; return;