Paper/src/main/java/net/minecraft/server/PlayerInteractManager.java

412 Zeilen
17 KiB
Java

2011-01-10 02:36:19 +01:00
package net.minecraft.server;
2011-02-01 18:06:17 +01:00
// CraftBukkit start
import org.bukkit.event.block.BlockBreakEvent;
import org.bukkit.craftbukkit.event.CraftEventFactory;
import org.bukkit.event.Event;
import org.bukkit.event.block.Action;
import org.bukkit.event.player.PlayerInteractEvent;
2011-02-01 18:06:17 +01:00
// CraftBukkit end
2012-12-20 05:03:52 +01:00
public class PlayerInteractManager {
2011-01-10 02:36:19 +01:00
2011-09-15 02:23:52 +02:00
public World world;
2012-07-29 09:33:13 +02:00
public EntityPlayer player;
private EnumGamemode gamemode;
private boolean d;
private int lastDigTick;
2011-03-31 22:40:00 +02:00
private int f;
2011-01-10 09:30:34 +01:00
private int g;
2011-09-15 02:23:52 +02:00
private int h;
private int currentTick;
2011-09-15 02:23:52 +02:00
private boolean j;
2011-03-31 22:40:00 +02:00
private int k;
2011-02-23 03:37:56 +01:00
private int l;
private int m;
2011-09-15 02:23:52 +02:00
private int n;
2012-07-29 09:33:13 +02:00
private int o;
2011-01-10 02:36:19 +01:00
2012-12-20 05:03:52 +01:00
public PlayerInteractManager(World world) {
2012-07-29 09:33:13 +02:00
this.gamemode = EnumGamemode.NONE;
this.o = -1;
2011-09-15 02:23:52 +02:00
this.world = world;
}
2012-07-29 09:33:13 +02:00
public void setGameMode(EnumGamemode enumgamemode) {
this.gamemode = enumgamemode;
enumgamemode.a(this.player.abilities);
this.player.updateAbilities();
2011-09-15 02:23:52 +02:00
}
2012-07-29 09:33:13 +02:00
public EnumGamemode getGameMode() {
return this.gamemode;
2011-09-15 02:23:52 +02:00
}
public boolean isCreative() {
2012-07-29 09:33:13 +02:00
return this.gamemode.d();
2011-09-15 02:23:52 +02:00
}
2012-07-29 09:33:13 +02:00
public void b(EnumGamemode enumgamemode) {
if (this.gamemode == EnumGamemode.NONE) {
this.gamemode = enumgamemode;
2011-09-15 02:23:52 +02:00
}
2012-07-29 09:33:13 +02:00
this.setGameMode(this.gamemode);
2011-01-10 02:36:19 +01:00
}
2012-07-29 09:33:13 +02:00
public void a() {
this.currentTick = MinecraftServer.currentTick; // CraftBukkit
2012-07-29 09:33:13 +02:00
float f;
2013-11-04 14:07:38 +01:00
int i;
2012-07-29 09:33:13 +02:00
2011-09-15 02:23:52 +02:00
if (this.j) {
2013-11-04 14:07:38 +01:00
int j = this.currentTick - this.n;
Block block = this.world.getType(this.k, this.l, this.m);
2011-02-23 03:37:56 +01:00
2013-11-04 14:07:38 +01:00
if (block.getMaterial() == Material.AIR) {
2012-07-29 09:33:13 +02:00
this.j = false;
} else {
2013-11-04 14:07:38 +01:00
f = block.getDamage(this.player, this.player.world, this.k, this.l, this.m) * (float) (j + 1);
i = (int) (f * 10.0F);
if (i != this.o) {
this.world.d(this.player.getId(), this.k, this.l, this.m, i);
this.o = i;
2012-07-29 09:33:13 +02:00
}
2011-02-23 03:37:56 +01:00
if (f >= 1.0F) {
2011-09-15 02:23:52 +02:00
this.j = false;
this.breakBlock(this.k, this.l, this.m);
2011-02-23 03:37:56 +01:00
}
2012-07-29 09:33:13 +02:00
}
} else if (this.d) {
2013-11-04 14:07:38 +01:00
Block block1 = this.world.getType(this.f, this.g, this.h);
2012-07-29 09:33:13 +02:00
2013-11-04 14:07:38 +01:00
if (block1.getMaterial() == Material.AIR) {
this.world.d(this.player.getId(), this.f, this.g, this.h, -1);
2012-07-29 09:33:13 +02:00
this.o = -1;
this.d = false;
2011-02-23 03:37:56 +01:00
} else {
2013-11-04 14:07:38 +01:00
int k = this.currentTick - this.lastDigTick;
2012-07-29 09:33:13 +02:00
2013-11-04 14:07:38 +01:00
f = block1.getDamage(this.player, this.player.world, this.f, this.g, this.h) * (float) (k + 1);
i = (int) (f * 10.0F);
if (i != this.o) {
this.world.d(this.player.getId(), this.f, this.g, this.h, i);
this.o = i;
2012-07-29 09:33:13 +02:00
}
2011-02-23 03:37:56 +01:00
}
}
}
2011-05-26 14:48:22 +02:00
public void dig(int i, int j, int k, int l) {
2013-11-04 14:07:38 +01:00
// CraftBukkit start
PlayerInteractEvent event = CraftEventFactory.callPlayerInteractEvent(this.player, Action.LEFT_CLICK_BLOCK, i, j, k, l, this.player.inventory.getItemInHand());
2013-07-01 13:03:00 +02:00
if (!this.gamemode.isAdventure() || this.player.d(i, j, k)) {
if (event.isCancelled()) {
// Let the client know the block still exists
2013-11-04 14:07:38 +01:00
((EntityPlayer) this.player).playerConnection.sendPacket(new PacketPlayOutBlockChange(i, j, k, this.world));
// Update any tile entity data for this block
TileEntity tileentity = this.world.getTileEntity(i, j, k);
if (tileentity != null) {
2012-12-20 05:03:52 +01:00
this.player.playerConnection.sendPacket(tileentity.getUpdatePacket());
}
return;
}
// CraftBukkit end
2012-07-29 09:33:13 +02:00
if (this.isCreative()) {
if (!this.world.douseFire((EntityHuman) null, i, j, k, l)) {
this.breakBlock(i, j, k);
}
} else {
2013-11-04 14:07:38 +01:00
// this.world.douseFire((EntityHuman) null, i, j, k, l); // CraftBukkit - Moved down
2012-07-29 09:33:13 +02:00
this.lastDigTick = this.currentTick;
float f = 1.0F;
2013-11-04 14:07:38 +01:00
Block block = this.world.getType(i, j, k);
2012-07-29 09:33:13 +02:00
// CraftBukkit start - Swings at air do *NOT* exist.
if (event.useInteractedBlock() == Event.Result.DENY) {
// If we denied a door from opening, we need to send a correcting update to the client, as it already opened the door.
2013-11-04 14:07:38 +01:00
if (block == Blocks.WOODEN_DOOR) {
2012-07-29 09:33:13 +02:00
// For some reason *BOTH* the bottom/top part have to be marked updated.
boolean bottom = (this.world.getData(i, j, k) & 8) == 0;
2013-11-04 14:07:38 +01:00
((EntityPlayer) this.player).playerConnection.sendPacket(new PacketPlayOutBlockChange(i, j, k, this.world));
((EntityPlayer) this.player).playerConnection.sendPacket(new PacketPlayOutBlockChange(i, j + (bottom ? 1 : -1), k, this.world));
} else if (block == Blocks.TRAP_DOOR) {
((EntityPlayer) this.player).playerConnection.sendPacket(new PacketPlayOutBlockChange(i, j, k, this.world));
2012-07-29 09:33:13 +02:00
}
2013-11-04 14:07:38 +01:00
} else if (block.getMaterial() != Material.AIR) {
block.attack(this.world, i, j, k, this.player);
f = block.getDamage(this.player, this.player.world, i, j, k);
2012-07-29 09:33:13 +02:00
// Allow fire punching to be blocked
this.world.douseFire((EntityHuman) null, i, j, k, l);
}
2012-07-29 09:33:13 +02:00
if (event.useItemInHand() == Event.Result.DENY) {
// If we 'insta destroyed' then the client needs to be informed.
if (f > 1.0f) {
2013-11-04 14:07:38 +01:00
((EntityPlayer) this.player).playerConnection.sendPacket(new PacketPlayOutBlockChange(i, j, k, this.world));
2012-07-29 09:33:13 +02:00
}
return;
2011-09-15 02:23:52 +02:00
}
org.bukkit.event.block.BlockDamageEvent blockEvent = CraftEventFactory.callBlockDamageEvent(this.player, i, j, k, this.player.inventory.getItemInHand(), f >= 1.0f);
2011-01-29 22:50:29 +01:00
2012-07-29 09:33:13 +02:00
if (blockEvent.isCancelled()) {
// Let the client know the block still exists
2013-11-04 14:07:38 +01:00
((EntityPlayer) this.player).playerConnection.sendPacket(new PacketPlayOutBlockChange(i, j, k, this.world));
2012-07-29 09:33:13 +02:00
return;
2011-09-15 02:23:52 +02:00
}
2011-03-23 16:38:42 +01:00
2012-07-29 09:33:13 +02:00
if (blockEvent.getInstaBreak()) {
f = 2.0f;
2012-07-29 09:33:13 +02:00
}
// CraftBukkit end
2011-03-23 16:38:42 +01:00
2013-11-04 14:07:38 +01:00
if (block.getMaterial() != Material.AIR && f >= 1.0F) {
2012-07-29 09:33:13 +02:00
this.breakBlock(i, j, k);
} else {
this.d = true;
this.f = i;
this.g = j;
this.h = k;
2013-11-04 14:07:38 +01:00
int i1 = (int) (f * 10.0F);
2012-07-29 09:33:13 +02:00
2013-11-04 14:07:38 +01:00
this.world.d(this.player.getId(), i, j, k, i1);
this.o = i1;
2012-07-29 09:33:13 +02:00
}
2011-09-15 02:23:52 +02:00
}
2011-01-10 02:36:19 +01:00
}
}
2011-05-26 14:48:22 +02:00
public void a(int i, int j, int k) {
2011-09-15 02:23:52 +02:00
if (i == this.f && j == this.g && k == this.h) {
this.currentTick = MinecraftServer.currentTick; // CraftBukkit
int l = this.currentTick - this.lastDigTick;
2013-11-04 14:07:38 +01:00
Block block = this.world.getType(i, j, k);
2011-01-29 22:50:29 +01:00
2013-11-04 14:07:38 +01:00
if (block.getMaterial() != Material.AIR) {
2012-07-29 09:33:13 +02:00
float f = block.getDamage(this.player, this.player.world, i, j, k) * (float) (l + 1);
2011-02-23 03:37:56 +01:00
2011-04-20 22:47:26 +02:00
if (f >= 0.7F) {
2012-07-29 09:33:13 +02:00
this.d = false;
2013-11-04 14:07:38 +01:00
this.world.d(this.player.getId(), i, j, k, -1);
this.breakBlock(i, j, k);
2011-09-15 02:23:52 +02:00
} else if (!this.j) {
2012-07-29 09:33:13 +02:00
this.d = false;
2011-09-15 02:23:52 +02:00
this.j = true;
this.k = i;
this.l = j;
this.m = k;
this.n = this.lastDigTick;
2011-01-29 22:50:29 +01:00
}
}
2013-03-25 05:22:32 +01:00
// CraftBukkit start - Force block reset to client
2011-03-24 22:42:23 +01:00
} else {
2013-11-04 14:07:38 +01:00
this.player.playerConnection.sendPacket(new PacketPlayOutBlockChange(i, j, k, this.world));
2011-03-24 22:42:23 +01:00
// CraftBukkit end
2011-01-10 02:36:19 +01:00
}
2012-07-29 09:33:13 +02:00
}
2011-02-23 03:37:56 +01:00
2012-07-29 09:33:13 +02:00
public void c(int i, int j, int k) {
this.d = false;
2013-11-04 14:07:38 +01:00
this.world.d(this.player.getId(), this.f, this.g, this.h, -1);
2011-01-10 02:36:19 +01:00
}
2012-07-29 09:33:13 +02:00
private boolean d(int i, int j, int k) {
2013-11-04 14:07:38 +01:00
Block block = this.world.getType(i, j, k);
int l = this.world.getData(i, j, k);
2012-07-29 09:33:13 +02:00
2013-11-04 14:07:38 +01:00
block.a(this.world, i, j, k, l, this.player);
2013-03-13 23:33:27 +01:00
boolean flag = this.world.setAir(i, j, k);
2011-01-10 02:36:19 +01:00
2013-11-04 14:07:38 +01:00
if (flag) {
block.postBreak(this.world, i, j, k, l);
2011-01-10 02:36:19 +01:00
}
2011-01-29 22:50:29 +01:00
2011-01-10 02:36:19 +01:00
return flag;
}
public boolean breakBlock(int i, int j, int k) {
// CraftBukkit start
BlockBreakEvent event = null;
if (this.player instanceof EntityPlayer) {
org.bukkit.block.Block block = this.world.getWorld().getBlockAt(i, j, k);
// Tell client the block is gone immediately then process events
if (world.getTileEntity(i, j, k) == null) {
2013-11-04 14:07:38 +01:00
PacketPlayOutBlockChange packet = new PacketPlayOutBlockChange(i, j, k, this.world);
packet.block = Blocks.AIR;
packet.data = 0;
2012-12-20 05:03:52 +01:00
((EntityPlayer) this.player).playerConnection.sendPacket(packet);
}
event = new BlockBreakEvent(block, this.player.getBukkitEntity());
// Adventure mode pre-cancel
2013-07-01 13:03:00 +02:00
event.setCancelled(this.gamemode.isAdventure() && !this.player.d(i, j, k));
// Sword + Creative mode pre-cancel
2013-11-04 14:07:38 +01:00
event.setCancelled(event.isCancelled() || (this.gamemode.d() && this.player.be() != null && this.player.be().getItem() instanceof ItemSword));
// Calculate default block experience
2013-11-04 14:07:38 +01:00
Block nmsBlock = this.world.getType(i, j, k);
2013-03-13 23:33:27 +01:00
if (nmsBlock != null && !event.isCancelled() && !this.isCreative() && this.player.a(nmsBlock)) {
2013-11-04 14:07:38 +01:00
// Copied from block.a(world, entityhuman, int, int, int, int)
if (!(nmsBlock.E() && EnchantmentManager.hasSilkTouchEnchantment(this.player))) {
int data = block.getData();
int bonusLevel = EnchantmentManager.getBonusBlockLootEnchantmentLevel(this.player);
event.setExpToDrop(nmsBlock.getExpDrop(this.world, data, bonusLevel));
}
}
this.world.getServer().getPluginManager().callEvent(event);
if (event.isCancelled()) {
// Let the client know the block still exists
2013-11-04 14:07:38 +01:00
((EntityPlayer) this.player).playerConnection.sendPacket(new PacketPlayOutBlockChange(i, j, k, this.world));
// Update any tile entity data for this block
TileEntity tileentity = this.world.getTileEntity(i, j, k);
if (tileentity != null) {
2012-12-20 05:03:52 +01:00
this.player.playerConnection.sendPacket(tileentity.getUpdatePacket());
}
2011-02-25 04:37:47 +01:00
return false;
2011-02-01 18:06:17 +01:00
}
}
2013-11-04 14:07:38 +01:00
if (false && this.gamemode.isAdventure() && !this.player.d(i, j, k)) { // Never trigger
2012-07-29 09:33:13 +02:00
// CraftBukkit end
return false;
2013-11-04 14:07:38 +01:00
} else if (false && this.gamemode.d() && this.player.be() != null && this.player.be().getItem() instanceof ItemSword) { // CraftBukkit - never trigger
return false;
2012-07-29 09:33:13 +02:00
} else {
2013-11-04 14:07:38 +01:00
Block block = this.world.getType(i, j, k);
if (block == Blocks.AIR) return false; // CraftBukkit - A plugin set block to air without cancelling
int l = this.world.getData(i, j, k);
2013-03-25 05:22:32 +01:00
// CraftBukkit start - Special case skulls, their item data comes from a tile entity
2013-11-04 14:07:38 +01:00
if (block == Blocks.SKULL && !this.isCreative()) {
block.dropNaturally(world, i, j, k, l, 1.0F, 0);
return this.d(i, j, k);
}
// CraftBukkit end
2013-11-04 14:07:38 +01:00
this.world.a(this.player, 2001, i, j, k, Block.b(block) + (this.world.getData(i, j, k) << 12));
2012-07-29 09:33:13 +02:00
boolean flag = this.d(i, j, k);
2011-01-10 02:36:19 +01:00
2012-07-29 09:33:13 +02:00
if (this.isCreative()) {
2013-11-04 14:07:38 +01:00
this.player.playerConnection.sendPacket(new PacketPlayOutBlockChange(i, j, k, this.world));
2012-07-29 09:33:13 +02:00
} else {
2013-11-04 14:07:38 +01:00
ItemStack itemstack = this.player.bD();
boolean flag1 = this.player.a(block);
2012-07-29 09:33:13 +02:00
if (itemstack != null) {
2013-11-04 14:07:38 +01:00
itemstack.a(this.world, block, i, j, k, this.player);
2012-07-29 09:33:13 +02:00
if (itemstack.count == 0) {
2013-11-04 14:07:38 +01:00
this.player.bE();
2012-07-29 09:33:13 +02:00
}
2011-09-15 02:23:52 +02:00
}
2011-01-29 22:50:29 +01:00
2012-07-29 09:33:13 +02:00
if (flag && flag1) {
2013-11-04 14:07:38 +01:00
block.a(this.world, this.player, i, j, k, l);
2012-07-29 09:33:13 +02:00
}
2011-09-15 02:23:52 +02:00
}
2011-01-29 22:50:29 +01:00
2013-03-25 05:22:32 +01:00
// CraftBukkit start - Drop event experience
if (flag && event != null) {
2013-11-04 14:07:38 +01:00
block.c(this.world, i, j, k, event.getExpToDrop());
}
// CraftBukkit end
2012-07-29 09:33:13 +02:00
return flag;
}
2011-01-10 02:36:19 +01:00
}
public boolean useItem(EntityHuman entityhuman, World world, ItemStack itemstack) {
2011-01-29 22:50:29 +01:00
int i = itemstack.count;
2011-09-15 02:23:52 +02:00
int j = itemstack.getData();
2011-01-29 22:50:29 +01:00
ItemStack itemstack1 = itemstack.a(world, entityhuman);
2011-01-10 02:36:19 +01:00
2013-03-13 23:33:27 +01:00
if (itemstack1 == itemstack && (itemstack1 == null || itemstack1.count == i && itemstack1.n() <= 0 && itemstack1.getData() == j)) {
2011-01-29 22:50:29 +01:00
return false;
} else {
entityhuman.inventory.items[entityhuman.inventory.itemInHandIndex] = itemstack1;
if (this.isCreative()) {
2011-09-15 02:23:52 +02:00
itemstack1.count = i;
2013-03-13 23:33:27 +01:00
if (itemstack1.g()) {
itemstack1.setData(j);
}
2011-09-15 02:23:52 +02:00
}
2011-01-29 22:50:29 +01:00
if (itemstack1.count == 0) {
entityhuman.inventory.items[entityhuman.inventory.itemInHandIndex] = null;
2011-01-10 02:36:19 +01:00
}
2011-01-29 22:50:29 +01:00
2013-11-04 14:07:38 +01:00
if (!entityhuman.bw()) {
((EntityPlayer) entityhuman).updateInventory(entityhuman.defaultContainer);
}
2011-01-10 02:36:19 +01:00
return true;
}
}
2012-07-29 09:33:13 +02:00
public boolean interact(EntityHuman entityhuman, World world, ItemStack itemstack, int i, int j, int k, int l, float f, float f1, float f2) {
2013-11-04 14:07:38 +01:00
/* CraftBukkit start - whole method
if ((!entityhuman.isSneaking() || entityhuman.be() == null) && world.getType(i, j, k).interact(world, i, j, k, entityhuman, l, f, f1, f2)) {
return true;
} else if (itemstack == null) {
return false;
} else if (this.isCreative()) {
int i1 = itemstack.getData();
int j1 = itemstack.count;
boolean flag = itemstack.placeItem(entityhuman, world, i, j, k, l, f, f1, f2);
2011-01-10 02:36:19 +01:00
2013-11-04 14:07:38 +01:00
itemstack.setData(i1);
itemstack.count = j1;
return flag;
} else {
return itemstack.placeItem(entityhuman, world, i, j, k, l, f, f1, f2);
}
// Interract event */
Block block = world.getType(i, j, k);
boolean result = false;
2013-11-04 14:07:38 +01:00
if (block != Blocks.AIR) {
PlayerInteractEvent event = CraftEventFactory.callPlayerInteractEvent(entityhuman, Action.RIGHT_CLICK_BLOCK, i, j, k, l, itemstack);
2011-03-24 12:11:28 +01:00
if (event.useInteractedBlock() == Event.Result.DENY) {
// If we denied a door from opening, we need to send a correcting update to the client, as it already opened the door.
2013-11-04 14:07:38 +01:00
if (block == Blocks.WOODEN_DOOR) {
boolean bottom = (world.getData(i, j, k) & 8) == 0;
2013-11-04 14:07:38 +01:00
((EntityPlayer) entityhuman).playerConnection.sendPacket(new PacketPlayOutBlockChange(i, j + (bottom ? 1 : -1), k, world));
}
result = (event.useItemInHand() != Event.Result.ALLOW);
2012-12-20 05:03:52 +01:00
} else if (!entityhuman.isSneaking() || itemstack == null) {
2013-11-04 14:07:38 +01:00
result = block.interact(world, i, j, k, entityhuman, l, f, f1, f2);
}
if (itemstack != null && !result) {
int j1 = itemstack.getData();
int k1 = itemstack.count;
2012-07-29 09:33:13 +02:00
result = itemstack.placeItem(entityhuman, world, i, j, k, l, f, f1, f2);
// The item count should not decrement in Creative mode.
if (this.isCreative()) {
itemstack.setData(j1);
itemstack.count = k1;
}
}
2011-03-24 12:11:28 +01:00
// If we have 'true' and no explicit deny *or* an explicit allow -- run the item part of the hook
if (itemstack != null && ((!result && event.useItemInHand() != Event.Result.DENY) || event.useItemInHand() == Event.Result.ALLOW)) {
this.useItem(entityhuman, world, itemstack);
}
}
return result;
2011-05-14 16:29:42 +02:00
// CraftBukkit end
2011-01-10 02:36:19 +01:00
}
2011-09-15 02:23:52 +02:00
public void a(WorldServer worldserver) {
this.world = worldserver;
}
2011-01-10 09:30:34 +01:00
}