package net.minecraft.server; import java.util.Random; // CraftBukkit start import org.bukkit.block.BlockFace; import org.bukkit.craftbukkit.block.CraftBlock; import org.bukkit.event.Event.Type; import org.bukkit.event.block.BlockFromToEvent; // CraftBukkit end public class BlockFlowing extends BlockFluids { int a = 0; boolean[] b = new boolean[4]; int[] c = new int[4]; protected BlockFlowing(int i, Material material) { super(i, material); } private void i(World world, int i, int j, int k) { int l = world.getData(i, j, k); world.setTypeIdAndData(i, j, k, this.id + 1, l); world.b(i, j, k, i, j, k); world.g(i, j, k); } public void a(World world, int i, int j, int k, Random random) { // CraftBukkit start CraftBlock source = (CraftBlock) ((WorldServer) world).getWorld().getBlockAt(i, j, k); // CraftBukkit end int l = this.g(world, i, j, k); byte b0 = 1; if (this.material == Material.LAVA && !world.m.d) { b0 = 2; } boolean flag = true; int i1; if (l > 0) { byte b1 = -100; this.a = 0; int j1 = this.f(world, i - 1, j, k, b1); j1 = this.f(world, i + 1, j, k, j1); j1 = this.f(world, i, j, k - 1, j1); j1 = this.f(world, i, j, k + 1, j1); i1 = j1 + b0; if (i1 >= 8 || j1 < 0) { i1 = -1; } if (this.g(world, i, j + 1, k) >= 0) { int k1 = this.g(world, i, j + 1, k); if (k1 >= 8) { i1 = k1; } else { i1 = k1 + 8; } } if (this.a >= 2 && this.material == Material.WATER) { if (world.d(i, j - 1, k)) { i1 = 0; } else if (world.getMaterial(i, j - 1, k) == this.material && world.getData(i, j, k) == 0) { i1 = 0; } } if (this.material == Material.LAVA && l < 8 && i1 < 8 && i1 > l && random.nextInt(4) != 0) { i1 = l; flag = false; } if (i1 != l) { l = i1; if (i1 < 0) { world.e(i, j, k, 0); } else { world.c(i, j, k, i1); world.c(i, j, k, this.id, this.b()); world.h(i, j, k, this.id); } } else if (flag) { this.i(world, i, j, k); } } else { this.i(world, i, j, k); } if (this.l(world, i, j - 1, k)) { // CraftBukkit start // Craftbucket send "down" to the server BlockFromToEvent event = new BlockFromToEvent(Type.BLOCK_FLOW, source, BlockFace.DOWN); ((WorldServer) world).getServer().getPluginManager().callEvent(event); if (!event.isCancelled()) { if (l >= 8) { world.b(i, j - 1, k, this.id, l); } else { world.b(i, j - 1, k, this.id, l + 8); } } // CraftBukkit end } else if (l >= 0 && (l == 0 || this.k(world, i, j - 1, k))) { boolean[] aboolean = this.j(world, i, j, k); i1 = l + b0; if (l >= 8) { i1 = 1; } if (i1 >= 8) { return; } // CraftBukkit start - all four cardinal directions. Do not change the order! BlockFace[] faces = new BlockFace[]{ BlockFace.NORTH, BlockFace.SOUTH, BlockFace.EAST, BlockFace.WEST }; int index = 0; for (BlockFace currentFace: faces) { if (aboolean[index]) { BlockFromToEvent event = new BlockFromToEvent(Type.BLOCK_FLOW, source, currentFace); ((WorldServer) world).getServer().getPluginManager().callEvent(event); if (!event.isCancelled()) { this.g(world, i + currentFace.getModX(), j, k + currentFace.getModZ(), i1); } } index++; } // CraftBukkit end } } private void g(World world, int i, int j, int k, int l) { if (this.l(world, i, j, k)) { int i1 = world.getTypeId(i, j, k); if (i1 > 0) { if (this.material == Material.LAVA) { this.h(world, i, j, k); } else { Block.byId[i1].b_(world, i, j, k, world.getData(i, j, k)); } } world.b(i, j, k, this.id, l); } } private int b(World world, int i, int j, int k, int l, int i1) { int j1 = 1000; for (int k1 = 0; k1 < 4; ++k1) { if ((k1 != 0 || i1 != 1) && (k1 != 1 || i1 != 0) && (k1 != 2 || i1 != 3) && (k1 != 3 || i1 != 2)) { int l1 = i; int i2 = k; if (k1 == 0) { l1 = i - 1; } if (k1 == 1) { ++l1; } if (k1 == 2) { i2 = k - 1; } if (k1 == 3) { ++i2; } if (!this.k(world, l1, j, i2) && (world.getMaterial(l1, j, i2) != this.material || world.getData(l1, j, i2) != 0)) { if (!this.k(world, l1, j - 1, i2)) { return l; } if (l < 4) { int j2 = this.b(world, l1, j, i2, l + 1, k1); if (j2 < j1) { j1 = j2; } } } } } return j1; } private boolean[] j(World world, int i, int j, int k) { int l; int i1; for (l = 0; l < 4; ++l) { this.c[l] = 1000; i1 = i; int j1 = k; if (l == 0) { i1 = i - 1; } if (l == 1) { ++i1; } if (l == 2) { j1 = k - 1; } if (l == 3) { ++j1; } if (!this.k(world, i1, j, j1) && (world.getMaterial(i1, j, j1) != this.material || world.getData(i1, j, j1) != 0)) { if (!this.k(world, i1, j - 1, j1)) { this.c[l] = 0; } else { this.c[l] = this.b(world, i1, j, j1, 1, l); } } } l = this.c[0]; for (i1 = 1; i1 < 4; ++i1) { if (this.c[i1] < l) { l = this.c[i1]; } } for (i1 = 0; i1 < 4; ++i1) { this.b[i1] = this.c[i1] == l; } return this.b; } private boolean k(World world, int i, int j, int k) { int l = world.getTypeId(i, j, k); if (l != Block.WOODEN_DOOR.id && l != Block.IRON_DOOR_BLOCK.id && l != Block.SIGN_POST.id && l != Block.LADDER.id && l != Block.SUGAR_CANE_BLOCK.id) { if (l == 0) { return false; } else { Material material = Block.byId[l].material; return material.isBuildable(); } } else { return true; } } protected int f(World world, int i, int j, int k, int l) { int i1 = this.g(world, i, j, k); if (i1 < 0) { return l; } else { if (i1 == 0) { ++this.a; } if (i1 >= 8) { i1 = 0; } return l >= 0 && i1 >= l ? l : i1; } } private boolean l(World world, int i, int j, int k) { Material material = world.getMaterial(i, j, k); return material == this.material ? false : (material == Material.LAVA ? false : !this.k(world, i, j, k)); } public void e(World world, int i, int j, int k) { super.e(world, i, j, k); if (world.getTypeId(i, j, k) == this.id) { world.c(i, j, k, this.id, this.b()); } } }