package net.minecraft.server; import it.unimi.dsi.fastutil.longs.Long2ObjectMap; import it.unimi.dsi.fastutil.longs.Long2ObjectOpenHashMap; import it.unimi.dsi.fastutil.objects.ObjectIterator; import java.util.Random; // CraftBukkit start import org.bukkit.Location; import org.bukkit.event.entity.EntityPortalExitEvent; import org.bukkit.util.Vector; // CraftBukkit end public class PortalTravelAgent { protected final WorldServer world; // Paper - private -> protected private final Random b; private final Long2ObjectMap<PortalTravelAgent.ChunkCoordinatesPortal> c = new Long2ObjectOpenHashMap(4096); public PortalTravelAgent(WorldServer worldserver) { this.world = worldserver; this.b = new Random(worldserver.getSeed()); } public void a(Entity entity, float f) { if (this.world.worldProvider.getDimensionManager().getDimensionID() != 1) { if (!this.b(entity, f)) { this.a(entity); this.b(entity, f); } } else { int i = MathHelper.floor(entity.locX); int j = MathHelper.floor(entity.locY) - 1; int k = MathHelper.floor(entity.locZ); // CraftBukkit start - Modularize end portal creation BlockPosition created = this.createEndPortal(entity.locX, entity.locY, entity.locZ); entity.setPositionRotation(created.getX(), created.getY(), created.getZ(), entity.yaw, 0.0F); entity.motX = entity.motY = entity.motZ = 0.0D; } } // Split out from original a(Entity, double, double, double, float) method in order to enable being called from createPortal private BlockPosition createEndPortal(double x, double y, double z) { int i = MathHelper.floor(x); int j = MathHelper.floor(y) - 1; int k = MathHelper.floor(z); // CraftBukkit end byte b0 = 1; byte b1 = 0; for (int l = -2; l <= 2; ++l) { for (int i1 = -2; i1 <= 2; ++i1) { for (int j1 = -1; j1 < 3; ++j1) { int k1 = i + i1 * 1 + l * 0; int l1 = j + j1; int i2 = k + i1 * 0 - l * 1; boolean flag2 = j1 < 0; this.world.setTypeUpdate(new BlockPosition(k1, l1, i2), flag2 ? Blocks.OBSIDIAN.getBlockData() : Blocks.AIR.getBlockData()); } } } // CraftBukkit start return new BlockPosition(i, k, k); } // use logic based on creation to verify end portal private BlockPosition findEndPortal(BlockPosition portal) { int i = portal.getX(); int j = portal.getY() - 1; int k = portal.getZ(); byte b0 = 1; byte b1 = 0; for (int l = -2; l <= 2; ++l) { for (int i1 = -2; i1 <= 2; ++i1) { for (int j1 = -1; j1 < 3; ++j1) { int k1 = i + i1 * b0 + l * b1; int l1 = j + j1; int i2 = k + i1 * b1 - l * b0; boolean flag = j1 < 0; if (this.world.getType(new BlockPosition(k1, l1, i2)).getBlock() != (flag ? Blocks.OBSIDIAN : Blocks.AIR)) { return null; } } } } return new BlockPosition(i, j, k); } // CraftBukkit end public boolean b(Entity entity, float f) { // CraftBukkit start - Modularize portal search process and entity teleportation BlockPosition found = this.findPortal(entity.locX, entity.locY, entity.locZ, world.paperConfig.portalSearchRadius); // Paper - Configurable search radius if (found == null) { return false; } Location exit = new Location(this.world.getWorld(), found.getX(), found.getY(), found.getZ(), f, entity.pitch); Vector velocity = entity.getBukkitEntity().getVelocity(); this.adjustExit(entity, exit, velocity); entity.setPositionRotation(exit.getX(), exit.getY(), exit.getZ(), exit.getYaw(), exit.getPitch()); if (entity.motX != velocity.getX() || entity.motY != velocity.getY() || entity.motZ != velocity.getZ()) { entity.getBukkitEntity().setVelocity(velocity); } return true; } public BlockPosition findPortal(double x, double y, double z, int radius) { if (this.world.getWorld().getEnvironment() == org.bukkit.World.Environment.THE_END) { return this.findEndPortal(this.world.worldProvider.h()); } // CraftBukkit end double d0 = -1.0D; // CraftBukkit start int i = MathHelper.floor(x); int j = MathHelper.floor(z); // CraftBukkit end boolean flag1 = true; Object object = BlockPosition.ZERO; long k = ChunkCoordIntPair.chunkXZ2Int(i, j); if (this.c.containsKey(k)) { PortalTravelAgent.ChunkCoordinatesPortal portaltravelagent_chunkcoordinatesportal = this.c.get(k); d0 = 0.0D; object = portaltravelagent_chunkcoordinatesportal; portaltravelagent_chunkcoordinatesportal.b = this.world.getTime(); flag1 = false; } else { BlockPosition blockposition = new BlockPosition(x, y, z); // CraftBukkit for (int l = -radius; l <= radius; ++l) { BlockPosition blockposition1; for (int i1 = -radius; i1 <= radius; ++i1) { for (BlockPosition blockposition2 = blockposition.a(l, this.world.Z() - 1 - blockposition.getY(), i1); blockposition2.getY() >= 0; blockposition2 = blockposition1) { blockposition1 = blockposition2.down(); if (this.world.getType(blockposition2).getBlock() == Blocks.PORTAL) { for (blockposition1 = blockposition2.down(); this.world.getType(blockposition1).getBlock() == Blocks.PORTAL; blockposition1 = blockposition1.down()) { blockposition2 = blockposition1; } double d1 = blockposition2.n(blockposition); if (d0 < 0.0D || d1 < d0) { d0 = d1; object = blockposition2; } } } } } } if (d0 >= 0.0D) { if (flag1) { this.c.put(k, new PortalTravelAgent.ChunkCoordinatesPortal((BlockPosition) object, this.world.getTime())); } // CraftBukkit start - Move entity teleportation logic into exit return (BlockPosition) object; } else { return null; } } // Entity repositioning logic split out from original b method and combined with repositioning logic for The End from original a method public void adjustExit(Entity entity, Location position, Vector velocity) { Location from = position.clone(); Vector before = velocity.clone(); BlockPosition object = new BlockPosition(position.getBlockX(), position.getBlockY(), position.getBlockZ()); float f = position.getYaw(); if (this.world.getWorld().getEnvironment() == org.bukkit.World.Environment.THE_END || entity.getBukkitEntity().getWorld().getEnvironment() == org.bukkit.World.Environment.THE_END || entity.getPortalOffset() == null) { // entity.setPositionRotation((double) i, (double) j, (double) k, entity.yaw, 0.0F); // entity.motX = entity.motY = entity.motZ = 0.0D; position.setPitch(0.0F); velocity.setX(0); velocity.setY(0); velocity.setZ(0); } else { // CraftBukkit end double d2 = object.getX() + 0.5D; double d3 = object.getZ() + 0.5D; ShapeDetector.ShapeDetectorCollection shapedetector_shapedetectorcollection = Blocks.PORTAL.c(this.world, object); boolean flag2 = shapedetector_shapedetectorcollection.getFacing().e().c() == EnumDirection.EnumAxisDirection.NEGATIVE; double d4 = shapedetector_shapedetectorcollection.getFacing().k() == EnumDirection.EnumAxis.X ? (double) shapedetector_shapedetectorcollection.a().getZ() : (double) shapedetector_shapedetectorcollection.a().getX(); double d5 = shapedetector_shapedetectorcollection.a().getY() + 1 - entity.getPortalOffset().y * shapedetector_shapedetectorcollection.e(); if (flag2) { ++d4; } if (shapedetector_shapedetectorcollection.getFacing().k() == EnumDirection.EnumAxis.X) { d3 = d4 + (1.0D - entity.getPortalOffset().x) * shapedetector_shapedetectorcollection.d() * shapedetector_shapedetectorcollection.getFacing().e().c().a(); } else { d2 = d4 + (1.0D - entity.getPortalOffset().x) * shapedetector_shapedetectorcollection.d() * shapedetector_shapedetectorcollection.getFacing().e().c().a(); } float f1 = 0.0F; float f2 = 0.0F; float f3 = 0.0F; float f4 = 0.0F; if (shapedetector_shapedetectorcollection.getFacing().opposite() == entity.getPortalDirection()) { f1 = 1.0F; f2 = 1.0F; } else if (shapedetector_shapedetectorcollection.getFacing().opposite() == entity.getPortalDirection().opposite()) { f1 = -1.0F; f2 = -1.0F; } else if (shapedetector_shapedetectorcollection.getFacing().opposite() == entity.getPortalDirection().e()) { f3 = 1.0F; f4 = -1.0F; } else { f3 = -1.0F; f4 = 1.0F; } // CraftBukkit start double d6 = velocity.getX(); double d7 = velocity.getZ(); // CraftBukkit end // CraftBukkit start - Adjust position and velocity instances instead of entity velocity.setX(d6 * f1 + d7 * f4); velocity.setZ(d6 * f3 + d7 * f2); f = f - entity.getPortalDirection().opposite().get2DRotationValue() * 90 + shapedetector_shapedetectorcollection.getFacing().get2DRotationValue() * 90; // entity.setPositionRotation(d2, d5, d3, entity.yaw, entity.pitch); position.setX(d2); position.setY(d5); position.setZ(d3); position.setYaw(f); } EntityPortalExitEvent event = new EntityPortalExitEvent(entity.getBukkitEntity(), from, position, before, velocity); this.world.getServer().getPluginManager().callEvent(event); Location to = event.getTo(); if (event.isCancelled() || to == null || !entity.isAlive()) { position.setX(from.getX()); position.setY(from.getY()); position.setZ(from.getZ()); position.setYaw(from.getYaw()); position.setPitch(from.getPitch()); velocity.copy(before); } else { position.setX(to.getX()); position.setY(to.getY()); position.setZ(to.getZ()); position.setYaw(to.getYaw()); position.setPitch(to.getPitch()); velocity.copy(event.getAfter()); // event.getAfter() will never be null, as setAfter() will cause an NPE if null is passed in } // CraftBukkit end } public boolean a(Entity entity) { // CraftBukkit start - Allow for portal creation to be based on coordinates instead of entity return this.createPortal(entity.locX, entity.locY, entity.locZ, 16); } public boolean createPortal(double x, double y, double z, int b0) { if (this.world.getWorld().getEnvironment() == org.bukkit.World.Environment.THE_END) { createEndPortal(x, y, z); return true; } // CraftBukkit end double d0 = -1.0D; // CraftBukkit start int i = MathHelper.floor(x); int j = MathHelper.floor(y); int k = MathHelper.floor(z); // CraftBukkit end int l = i; int i1 = j; int j1 = k; int k1 = 0; int l1 = this.b.nextInt(4); BlockPosition.MutableBlockPosition blockposition_mutableblockposition = new BlockPosition.MutableBlockPosition(); int i2; double d1; int j2; double d2; int k2; int l2; int i3; int j3; int k3; int l3; int i4; int j4; int k4; double d3; double d4; for (i2 = i - 16; i2 <= i + 16; ++i2) { d1 = i2 + 0.5D - x; // CraftBukkit for (j2 = k - 16; j2 <= k + 16; ++j2) { d2 = j2 + 0.5D - z; // CraftBukkit label271: for (k2 = this.world.Z() - 1; k2 >= 0; --k2) { if (this.world.isEmpty(blockposition_mutableblockposition.c(i2, k2, j2))) { while (k2 > 0 && this.world.isEmpty(blockposition_mutableblockposition.c(i2, k2 - 1, j2))) { --k2; } for (l2 = l1; l2 < l1 + 4; ++l2) { i3 = l2 % 2; j3 = 1 - i3; if (l2 % 4 >= 2) { i3 = -i3; j3 = -j3; } for (k3 = 0; k3 < 3; ++k3) { for (l3 = 0; l3 < 4; ++l3) { for (i4 = -1; i4 < 4; ++i4) { j4 = i2 + (l3 - 1) * i3 + k3 * j3; k4 = k2 + i4; int l4 = j2 + (l3 - 1) * j3 - k3 * i3; blockposition_mutableblockposition.c(j4, k4, l4); if (i4 < 0 && !this.world.getType(blockposition_mutableblockposition).getMaterial().isBuildable() || i4 >= 0 && !this.world.isEmpty(blockposition_mutableblockposition)) { continue label271; } } } } d3 = k2 + 0.5D - y; // CraftBukkit d4 = d1 * d1 + d3 * d3 + d2 * d2; if (d0 < 0.0D || d4 < d0) { d0 = d4; l = i2; i1 = k2; j1 = j2; k1 = l2 % 4; } } } } } } if (d0 < 0.0D) { for (i2 = i - 16; i2 <= i + 16; ++i2) { d1 = i2 + 0.5D - x; // CraftBukkit for (j2 = k - 16; j2 <= k + 16; ++j2) { d2 = j2 + 0.5D - z; // CraftBukkit label219: for (k2 = this.world.Z() - 1; k2 >= 0; --k2) { if (this.world.isEmpty(blockposition_mutableblockposition.c(i2, k2, j2))) { while (k2 > 0 && this.world.isEmpty(blockposition_mutableblockposition.c(i2, k2 - 1, j2))) { --k2; } for (l2 = l1; l2 < l1 + 2; ++l2) { i3 = l2 % 2; j3 = 1 - i3; for (k3 = 0; k3 < 4; ++k3) { for (l3 = -1; l3 < 4; ++l3) { i4 = i2 + (k3 - 1) * i3; j4 = k2 + l3; k4 = j2 + (k3 - 1) * j3; blockposition_mutableblockposition.c(i4, j4, k4); if (l3 < 0 && !this.world.getType(blockposition_mutableblockposition).getMaterial().isBuildable() || l3 >= 0 && !this.world.isEmpty(blockposition_mutableblockposition)) { continue label219; } } } d3 = k2 + 0.5D - y; // CraftBukkit d4 = d1 * d1 + d3 * d3 + d2 * d2; if (d0 < 0.0D || d4 < d0) { d0 = d4; l = i2; i1 = k2; j1 = j2; k1 = l2 % 2; } } } } } } } int i5 = l; int j5 = i1; j2 = j1; int k5 = k1 % 2; int l5 = 1 - k5; if (k1 % 4 >= 2) { k5 = -k5; l5 = -l5; } if (d0 < 0.0D) { i1 = MathHelper.clamp(i1, 70, this.world.Z() - 10); j5 = i1; for (k2 = -1; k2 <= 1; ++k2) { for (l2 = 1; l2 < 3; ++l2) { for (i3 = -1; i3 < 3; ++i3) { j3 = i5 + (l2 - 1) * k5 + k2 * l5; k3 = j5 + i3; l3 = j2 + (l2 - 1) * l5 - k2 * k5; boolean flag1 = i3 < 0; this.world.setTypeUpdate(new BlockPosition(j3, k3, l3), flag1 ? Blocks.OBSIDIAN.getBlockData() : Blocks.AIR.getBlockData()); } } } } IBlockData iblockdata = Blocks.PORTAL.getBlockData().set(BlockPortal.AXIS, k5 == 0 ? EnumDirection.EnumAxis.Z : EnumDirection.EnumAxis.X); for (l2 = 0; l2 < 4; ++l2) { for (i3 = 0; i3 < 4; ++i3) { for (j3 = -1; j3 < 4; ++j3) { k3 = i5 + (i3 - 1) * k5; l3 = j5 + j3; i4 = j2 + (i3 - 1) * l5; boolean flag2 = i3 == 0 || i3 == 3 || j3 == -1 || j3 == 3; this.world.setTypeAndData(new BlockPosition(k3, l3, i4), flag2 ? Blocks.OBSIDIAN.getBlockData() : iblockdata, 2); } } for (i3 = 0; i3 < 4; ++i3) { for (j3 = -1; j3 < 4; ++j3) { k3 = i5 + (i3 - 1) * k5; l3 = j5 + j3; i4 = j2 + (i3 - 1) * l5; BlockPosition blockposition = new BlockPosition(k3, l3, i4); this.world.applyPhysics(blockposition, this.world.getType(blockposition).getBlock(), false); } } } return true; } public void a(long i) { if (i % 100L == 0L) { long j = i - 300L; ObjectIterator objectiterator = this.c.values().iterator(); while (objectiterator.hasNext()) { PortalTravelAgent.ChunkCoordinatesPortal portaltravelagent_chunkcoordinatesportal = (PortalTravelAgent.ChunkCoordinatesPortal) objectiterator.next(); if (portaltravelagent_chunkcoordinatesportal == null || portaltravelagent_chunkcoordinatesportal.b < j) { objectiterator.remove(); } } } } public class ChunkCoordinatesPortal extends BlockPosition { public long b; public ChunkCoordinatesPortal(BlockPosition blockposition, long i) { super(blockposition.getX(), blockposition.getY(), blockposition.getZ()); this.b = i; } @Override public int compareTo(BaseBlockPosition o) { return this.l(o); } } }