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);
}
}
}