package net.citizensnpcs.nms.v1_16_R3.entity; import org.bukkit.Bukkit; import org.bukkit.craftbukkit.v1_16_R3.CraftServer; import org.bukkit.craftbukkit.v1_16_R3.entity.CraftEntity; import org.bukkit.craftbukkit.v1_16_R3.entity.CraftPhantom; import org.bukkit.entity.Phantom; import org.bukkit.util.Vector; import net.citizensnpcs.api.npc.NPC; import net.citizensnpcs.nms.v1_16_R3.util.ForwardingNPCHolder; import net.citizensnpcs.nms.v1_16_R3.util.NMSBoundingBox; import net.citizensnpcs.nms.v1_16_R3.util.NMSImpl; import net.citizensnpcs.nms.v1_16_R3.util.EntityMoveControl; import net.citizensnpcs.npc.CitizensNPC; import net.citizensnpcs.npc.ai.NPCHolder; import net.citizensnpcs.util.NMS; import net.citizensnpcs.util.Util; import net.minecraft.server.v1_16_R3.AxisAlignedBB; import net.minecraft.server.v1_16_R3.BlockPosition; import net.minecraft.server.v1_16_R3.ControllerLook; import net.minecraft.server.v1_16_R3.ControllerMove; import net.minecraft.server.v1_16_R3.DamageSource; import net.minecraft.server.v1_16_R3.DataWatcherObject; import net.minecraft.server.v1_16_R3.Entity; import net.minecraft.server.v1_16_R3.EntityBoat; import net.minecraft.server.v1_16_R3.EntityMinecartAbstract; import net.minecraft.server.v1_16_R3.EntityPhantom; import net.minecraft.server.v1_16_R3.EntityTypes; import net.minecraft.server.v1_16_R3.FluidType; import net.minecraft.server.v1_16_R3.IBlockData; import net.minecraft.server.v1_16_R3.NBTTagCompound; import net.minecraft.server.v1_16_R3.SoundEffect; import net.minecraft.server.v1_16_R3.Tag; import net.minecraft.server.v1_16_R3.Vec3D; import net.minecraft.server.v1_16_R3.World; public class PhantomController extends MobEntityController { public PhantomController() { super(EntityPhantomNPC.class); } @Override public Phantom getBukkitEntity() { return (Phantom) super.getBukkitEntity(); } public static class EntityPhantomNPC extends EntityPhantom implements NPCHolder { private final CitizensNPC npc; private ControllerLook oldLookController; private ControllerMove oldMoveController; public EntityPhantomNPC(EntityTypes types, World world) { this(types, world, null); } public EntityPhantomNPC(EntityTypes types, World world, NPC npc) { super(types, world); this.npc = (CitizensNPC) npc; if (npc != null) { this.oldMoveController = this.moveController; this.oldLookController = this.lookController; this.moveController = new ControllerMove(this); this.lookController = new ControllerLook(this); // TODO: phantom pitch reversed } } @Override public void a(AxisAlignedBB bb) { super.a(NMSBoundingBox.makeBB(npc, bb)); } @Override public void a(DataWatcherObject datawatcherobject) { if (npc == null) { super.a(datawatcherobject); return; } NMSImpl.checkAndUpdateHeight(this, datawatcherobject, super::a); } @Override protected void a(double d0, boolean flag, IBlockData block, BlockPosition blockposition) { if (npc == null || !npc.isFlyable()) { super.a(d0, flag, block, blockposition); } } @Override public void a(float strength, double dx, double dz) { NMS.callKnockbackEvent(npc, strength, dx, dz, (evt) -> super.a((float) evt.getStrength(), evt.getKnockbackVector().getX(), evt.getKnockbackVector().getZ())); } @Override public boolean a(Tag tag, double d0) { return NMSImpl.fluidPush(npc, this, () -> super.a(tag, d0)); } @Override public boolean b(float f, float f1) { if (npc == null || !npc.isFlyable()) { return super.b(f, f1); } return false; } @Override public int bP() { return NMS.getFallDistance(npc, super.bP()); } @Override public void checkDespawn() { if (npc == null) { super.checkDespawn(); } } @Override public void collide(net.minecraft.server.v1_16_R3.Entity entity) { // this method is called by both the entities involved - cancelling // it will not stop the NPC from moving. super.collide(entity); if (npc != null) Util.callCollisionEvent(npc, entity.getBukkitEntity()); } @Override public boolean d(NBTTagCompound save) { return npc == null ? super.d(save) : false; } @Override public boolean eG() { if (npc == null || !npc.isProtected()) return super.eG(); return false; } @Override public void g(Vec3D vec3d) { if (npc == null || !npc.isFlyable()) { super.g(vec3d); } else { NMSImpl.flyingMoveLogic(this, vec3d); } } @Override public CraftEntity getBukkitEntity() { if (npc != null && !(super.getBukkitEntity() instanceof NPCHolder)) { NMSImpl.setBukkitEntity(this, new PhantomNPC(this)); } return super.getBukkitEntity(); } @Override public NPC getNPC() { return npc; } @Override protected SoundEffect getSoundAmbient() { return NMSImpl.getSoundEffect(npc, super.getSoundAmbient(), NPC.Metadata.AMBIENT_SOUND); } @Override protected SoundEffect getSoundDeath() { return NMSImpl.getSoundEffect(npc, super.getSoundDeath(), NPC.Metadata.DEATH_SOUND); } @Override protected SoundEffect getSoundHurt(DamageSource damagesource) { return NMSImpl.getSoundEffect(npc, super.getSoundHurt(damagesource), NPC.Metadata.HURT_SOUND); } @Override public void i(double x, double y, double z) { Vector vector = Util.callPushEvent(npc, x, y, z); if (vector != null) { super.i(vector.getX(), vector.getY(), vector.getZ()); } } @Override public boolean isClimbing() { if (npc == null || !npc.isFlyable()) { return super.isClimbing(); } else { return false; } } @Override public boolean isLeashed() { return NMSImpl.isLeashed(npc, super::isLeashed, this); } @Override protected boolean L() { return npc != null ? false : super.L(); } @Override public void movementTick() { super.movementTick(); if (npc != null) { NMSImpl.updateMinecraftAIState(npc, this); if (npc.useMinecraftAI() && this.moveController != this.oldMoveController) { this.moveController = this.oldMoveController; this.lookController = this.oldLookController; } if (!npc.useMinecraftAI() && this.moveController == this.oldMoveController) { this.moveController = new EntityMoveControl(this); this.lookController = new ControllerLook(this); } if (npc.isProtected()) { setOnFire(0); } npc.update(); } } @Override protected boolean n(Entity entity) { if (npc != null && (entity instanceof EntityBoat || entity instanceof EntityMinecartAbstract)) { return !npc.isProtected(); } return super.n(entity); } } public static class PhantomNPC extends CraftPhantom implements ForwardingNPCHolder { public PhantomNPC(EntityPhantomNPC entity) { super((CraftServer) Bukkit.getServer(), entity); } } }