1622 lines
46 KiB
Java
1622 lines
46 KiB
Java
package net.minecraft.entity;
|
|
|
|
import java.util.Collection;
|
|
import java.util.Iterator;
|
|
import java.util.List;
|
|
import java.util.Map;
|
|
import net.lax1dude.eaglercraft.v1_8.EaglercraftRandom;
|
|
import net.lax1dude.eaglercraft.v1_8.EaglercraftUUID;
|
|
|
|
import com.google.common.base.Predicate;
|
|
import com.google.common.base.Predicates;
|
|
import com.google.common.collect.Maps;
|
|
|
|
import net.minecraft.block.Block;
|
|
import net.minecraft.block.material.Material;
|
|
import net.minecraft.enchantment.EnchantmentHelper;
|
|
import net.minecraft.entity.ai.attributes.AttributeModifier;
|
|
import net.minecraft.entity.ai.attributes.BaseAttributeMap;
|
|
import net.minecraft.entity.ai.attributes.IAttribute;
|
|
import net.minecraft.entity.ai.attributes.IAttributeInstance;
|
|
import net.minecraft.entity.ai.attributes.ServersideAttributeMap;
|
|
import net.minecraft.entity.player.EntityPlayer;
|
|
import net.minecraft.init.Blocks;
|
|
import net.minecraft.item.Item;
|
|
import net.minecraft.item.ItemArmor;
|
|
import net.minecraft.item.ItemStack;
|
|
import net.minecraft.nbt.NBTBase;
|
|
import net.minecraft.nbt.NBTTagCompound;
|
|
import net.minecraft.nbt.NBTTagFloat;
|
|
import net.minecraft.nbt.NBTTagList;
|
|
import net.minecraft.nbt.NBTTagShort;
|
|
import net.minecraft.potion.Potion;
|
|
import net.minecraft.potion.PotionEffect;
|
|
import net.minecraft.potion.PotionHelper;
|
|
import net.minecraft.scoreboard.Team;
|
|
import net.minecraft.util.AxisAlignedBB;
|
|
import net.minecraft.util.BlockPos;
|
|
import net.minecraft.util.CombatTracker;
|
|
import net.minecraft.util.DamageSource;
|
|
import net.minecraft.util.EntitySelectors;
|
|
import net.minecraft.util.EnumParticleTypes;
|
|
import net.minecraft.util.MathHelper;
|
|
import net.minecraft.util.Vec3;
|
|
import net.minecraft.world.World;
|
|
|
|
public abstract class EntityLivingBase extends Entity {
|
|
private static final EaglercraftUUID sprintingSpeedBoostModifierUUID = EaglercraftUUID
|
|
.fromString("662A6B8D-DA3E-4C1C-8813-96EA6097278D");
|
|
private static final AttributeModifier sprintingSpeedBoostModifier = (new AttributeModifier(
|
|
sprintingSpeedBoostModifierUUID, "Sprinting speed boost", 0.30000001192092896D, 2)).setSaved(false);
|
|
private BaseAttributeMap attributeMap;
|
|
private final CombatTracker _combatTracker = new CombatTracker(this);
|
|
private final Map<Integer, PotionEffect> activePotionsMap = Maps.newHashMap();
|
|
/**+
|
|
* The equipment this mob was previously wearing, used for
|
|
* syncing.
|
|
*/
|
|
private final ItemStack[] previousEquipment = new ItemStack[5];
|
|
public boolean isSwingInProgress;
|
|
public int swingProgressInt;
|
|
public int arrowHitTimer;
|
|
public int hurtTime;
|
|
public int maxHurtTime;
|
|
public float attackedAtYaw;
|
|
public int deathTime;
|
|
public float prevSwingProgress;
|
|
public float swingProgress;
|
|
public float prevLimbSwingAmount;
|
|
public float limbSwingAmount;
|
|
public float limbSwing;
|
|
public int maxHurtResistantTime = 20;
|
|
public float prevCameraPitch;
|
|
public float cameraPitch;
|
|
public float field_70769_ao;
|
|
public float field_70770_ap;
|
|
public float renderYawOffset;
|
|
public float prevRenderYawOffset;
|
|
public float rotationYawHead;
|
|
public float prevRotationYawHead;
|
|
/**+
|
|
* A factor used to determine how far this entity will move each
|
|
* tick if it is jumping or falling.
|
|
*/
|
|
public float jumpMovementFactor = 0.02F;
|
|
protected EntityPlayer attackingPlayer;
|
|
protected int recentlyHit;
|
|
protected boolean dead;
|
|
protected int entityAge;
|
|
protected float prevOnGroundSpeedFactor;
|
|
protected float onGroundSpeedFactor;
|
|
protected float movedDistance;
|
|
protected float prevMovedDistance;
|
|
protected float field_70741_aB;
|
|
protected int scoreValue;
|
|
protected float lastDamage;
|
|
protected boolean isJumping;
|
|
public float moveStrafing;
|
|
public float moveForward;
|
|
protected float randomYawVelocity;
|
|
protected int newPosRotationIncrements;
|
|
protected double newPosX;
|
|
protected double newPosY;
|
|
protected double newPosZ;
|
|
protected double newRotationYaw;
|
|
protected double newRotationPitch;
|
|
/**+
|
|
* Whether the DataWatcher needs to be updated with the active
|
|
* potions
|
|
*/
|
|
private boolean potionsNeedUpdate = true;
|
|
private EntityLivingBase entityLivingToAttack;
|
|
private int revengeTimer;
|
|
private EntityLivingBase lastAttacker;
|
|
private int lastAttackerTime;
|
|
private float landMovementFactor;
|
|
private int jumpTicks;
|
|
private float absorptionAmount;
|
|
|
|
/**+
|
|
* Called by the /kill command.
|
|
*/
|
|
public void onKillCommand() {
|
|
this.attackEntityFrom(DamageSource.outOfWorld, Float.MAX_VALUE);
|
|
}
|
|
|
|
public EntityLivingBase(World worldIn) {
|
|
super(worldIn);
|
|
this.applyEntityAttributes();
|
|
this.setHealth(this.getMaxHealth());
|
|
this.preventEntitySpawning = true;
|
|
this.field_70770_ap = (float) ((Math.random() + 1.0D) * 0.009999999776482582D);
|
|
this.setPosition(this.posX, this.posY, this.posZ);
|
|
this.field_70769_ao = (float) Math.random() * 12398.0F;
|
|
this.rotationYaw = (float) (Math.random() * 3.1415927410125732D * 2.0D);
|
|
this.rotationYawHead = this.rotationYaw;
|
|
this.stepHeight = 0.6F;
|
|
}
|
|
|
|
protected void entityInit() {
|
|
this.dataWatcher.addObject(7, Integer.valueOf(0));
|
|
this.dataWatcher.addObject(8, Byte.valueOf((byte) 0));
|
|
this.dataWatcher.addObject(9, Byte.valueOf((byte) 0));
|
|
this.dataWatcher.addObject(6, Float.valueOf(1.0F));
|
|
}
|
|
|
|
protected void applyEntityAttributes() {
|
|
this.getAttributeMap().registerAttribute(SharedMonsterAttributes.maxHealth);
|
|
this.getAttributeMap().registerAttribute(SharedMonsterAttributes.knockbackResistance);
|
|
this.getAttributeMap().registerAttribute(SharedMonsterAttributes.movementSpeed);
|
|
}
|
|
|
|
protected void updateFallState(double d0, boolean flag, Block block, BlockPos blockpos) {
|
|
if (!this.isInWater()) {
|
|
this.handleWaterMovement();
|
|
}
|
|
|
|
super.updateFallState(d0, flag, block, blockpos);
|
|
}
|
|
|
|
public boolean canBreatheUnderwater() {
|
|
return false;
|
|
}
|
|
|
|
/**+
|
|
* Gets called every tick from main Entity class
|
|
*/
|
|
public void onEntityUpdate() {
|
|
this.prevSwingProgress = this.swingProgress;
|
|
super.onEntityUpdate();
|
|
this.worldObj.theProfiler.startSection("livingEntityBaseTick");
|
|
boolean flag = this instanceof EntityPlayer;
|
|
if (this.isEntityAlive()) {
|
|
if (this.isEntityInsideOpaqueBlock()) {
|
|
this.attackEntityFrom(DamageSource.inWall, 1.0F);
|
|
} else if (flag && !this.worldObj.getWorldBorder().contains(this.getEntityBoundingBox())) {
|
|
double d0 = this.worldObj.getWorldBorder().getClosestDistance(this)
|
|
+ this.worldObj.getWorldBorder().getDamageBuffer();
|
|
if (d0 < 0.0D) {
|
|
this.attackEntityFrom(DamageSource.inWall, (float) Math.max(1,
|
|
MathHelper.floor_double(-d0 * this.worldObj.getWorldBorder().getDamageAmount())));
|
|
}
|
|
}
|
|
}
|
|
|
|
this.extinguish();
|
|
|
|
boolean flag1 = flag && ((EntityPlayer) this).capabilities.disableDamage;
|
|
if (this.isEntityAlive()) {
|
|
if (this.isInsideOfMaterial(Material.water)) {
|
|
if (!this.canBreatheUnderwater() && !this.isPotionActive(Potion.waterBreathing.id) && !flag1) {
|
|
this.setAir(this.decreaseAirSupply(this.getAir()));
|
|
if (this.getAir() == -20) {
|
|
this.setAir(0);
|
|
|
|
for (int i = 0; i < 8; ++i) {
|
|
float f = this.rand.nextFloat() - this.rand.nextFloat();
|
|
float f1 = this.rand.nextFloat() - this.rand.nextFloat();
|
|
float f2 = this.rand.nextFloat() - this.rand.nextFloat();
|
|
this.worldObj.spawnParticle(EnumParticleTypes.WATER_BUBBLE, this.posX + (double) f,
|
|
this.posY + (double) f1, this.posZ + (double) f2, this.motionX, this.motionY,
|
|
this.motionZ, new int[0]);
|
|
}
|
|
|
|
this.attackEntityFrom(DamageSource.drown, 2.0F);
|
|
}
|
|
}
|
|
|
|
} else {
|
|
this.setAir(300);
|
|
}
|
|
}
|
|
|
|
if (this.isEntityAlive() && this.isWet()) {
|
|
this.extinguish();
|
|
}
|
|
|
|
this.prevCameraPitch = this.cameraPitch;
|
|
if (this.hurtTime > 0) {
|
|
--this.hurtTime;
|
|
}
|
|
|
|
if (this.hurtResistantTime > 0) {
|
|
--this.hurtResistantTime;
|
|
}
|
|
|
|
if (this.getHealth() <= 0.0F) {
|
|
this.onDeathUpdate();
|
|
}
|
|
|
|
if (this.recentlyHit > 0) {
|
|
--this.recentlyHit;
|
|
} else {
|
|
this.attackingPlayer = null;
|
|
}
|
|
|
|
if (this.lastAttacker != null && !this.lastAttacker.isEntityAlive()) {
|
|
this.lastAttacker = null;
|
|
}
|
|
|
|
if (this.entityLivingToAttack != null) {
|
|
if (!this.entityLivingToAttack.isEntityAlive()) {
|
|
this.setRevengeTarget((EntityLivingBase) null);
|
|
} else if (this.ticksExisted - this.revengeTimer > 100) {
|
|
this.setRevengeTarget((EntityLivingBase) null);
|
|
}
|
|
}
|
|
|
|
this.updatePotionEffects();
|
|
this.prevMovedDistance = this.movedDistance;
|
|
this.prevRenderYawOffset = this.renderYawOffset;
|
|
this.prevRotationYawHead = this.rotationYawHead;
|
|
this.prevRotationYaw = this.rotationYaw;
|
|
this.prevRotationPitch = this.rotationPitch;
|
|
this.worldObj.theProfiler.endSection();
|
|
}
|
|
|
|
/**+
|
|
* If Animal, checks if the age timer is negative
|
|
*/
|
|
public boolean isChild() {
|
|
return false;
|
|
}
|
|
|
|
/**+
|
|
* handles entity death timer, experience orb and particle
|
|
* creation
|
|
*/
|
|
protected void onDeathUpdate() {
|
|
++this.deathTime;
|
|
if (this.deathTime == 20) {
|
|
|
|
this.setDead();
|
|
|
|
for (int k = 0; k < 20; ++k) {
|
|
double d2 = this.rand.nextGaussian() * 0.02D;
|
|
double d0 = this.rand.nextGaussian() * 0.02D;
|
|
double d1 = this.rand.nextGaussian() * 0.02D;
|
|
this.worldObj.spawnParticle(EnumParticleTypes.EXPLOSION_NORMAL,
|
|
this.posX + (double) (this.rand.nextFloat() * this.width * 2.0F) - (double) this.width,
|
|
this.posY + (double) (this.rand.nextFloat() * this.height),
|
|
this.posZ + (double) (this.rand.nextFloat() * this.width * 2.0F) - (double) this.width, d2, d0,
|
|
d1, new int[0]);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
/**+
|
|
* Entity won't drop items or experience points if this returns
|
|
* false
|
|
*/
|
|
protected boolean canDropLoot() {
|
|
return !this.isChild();
|
|
}
|
|
|
|
/**+
|
|
* Decrements the entity's air supply when underwater
|
|
*/
|
|
protected int decreaseAirSupply(int parInt1) {
|
|
int i = EnchantmentHelper.getRespiration(this);
|
|
return i > 0 && this.rand.nextInt(i + 1) > 0 ? parInt1 : parInt1 - 1;
|
|
}
|
|
|
|
/**+
|
|
* Get the experience points the entity currently has.
|
|
*/
|
|
protected int getExperiencePoints(EntityPlayer player) {
|
|
return 0;
|
|
}
|
|
|
|
/**+
|
|
* Only use is to identify if class is an instance of player for
|
|
* experience dropping
|
|
*/
|
|
protected boolean isPlayer() {
|
|
return false;
|
|
}
|
|
|
|
public EaglercraftRandom getRNG() {
|
|
return this.rand;
|
|
}
|
|
|
|
public EntityLivingBase getAITarget() {
|
|
return this.entityLivingToAttack;
|
|
}
|
|
|
|
public int getRevengeTimer() {
|
|
return this.revengeTimer;
|
|
}
|
|
|
|
public void setRevengeTarget(EntityLivingBase livingBase) {
|
|
this.entityLivingToAttack = livingBase;
|
|
this.revengeTimer = this.ticksExisted;
|
|
}
|
|
|
|
public EntityLivingBase getLastAttacker() {
|
|
return this.lastAttacker;
|
|
}
|
|
|
|
public int getLastAttackerTime() {
|
|
return this.lastAttackerTime;
|
|
}
|
|
|
|
public void setLastAttacker(Entity entityIn) {
|
|
if (entityIn instanceof EntityLivingBase) {
|
|
this.lastAttacker = (EntityLivingBase) entityIn;
|
|
} else {
|
|
this.lastAttacker = null;
|
|
}
|
|
|
|
this.lastAttackerTime = this.ticksExisted;
|
|
}
|
|
|
|
public int getAge() {
|
|
return this.entityAge;
|
|
}
|
|
|
|
/**+
|
|
* (abstract) Protected helper method to write subclass entity
|
|
* data to NBT.
|
|
*/
|
|
public void writeEntityToNBT(NBTTagCompound nbttagcompound) {
|
|
nbttagcompound.setFloat("HealF", this.getHealth());
|
|
nbttagcompound.setShort("Health", (short) ((int) Math.ceil((double) this.getHealth())));
|
|
nbttagcompound.setShort("HurtTime", (short) this.hurtTime);
|
|
nbttagcompound.setInteger("HurtByTimestamp", this.revengeTimer);
|
|
nbttagcompound.setShort("DeathTime", (short) this.deathTime);
|
|
nbttagcompound.setFloat("AbsorptionAmount", this.getAbsorptionAmount());
|
|
|
|
for (ItemStack itemstack : this.getInventory()) {
|
|
if (itemstack != null) {
|
|
this.attributeMap.removeAttributeModifiers(itemstack.getAttributeModifiers());
|
|
}
|
|
}
|
|
|
|
nbttagcompound.setTag("Attributes", SharedMonsterAttributes.writeBaseAttributeMapToNBT(this.getAttributeMap()));
|
|
|
|
for (ItemStack itemstack1 : this.getInventory()) {
|
|
if (itemstack1 != null) {
|
|
this.attributeMap.applyAttributeModifiers(itemstack1.getAttributeModifiers());
|
|
}
|
|
}
|
|
|
|
if (!this.activePotionsMap.isEmpty()) {
|
|
NBTTagList nbttaglist = new NBTTagList();
|
|
|
|
for (PotionEffect potioneffect : this.activePotionsMap.values()) {
|
|
nbttaglist.appendTag(potioneffect.writeCustomPotionEffectToNBT(new NBTTagCompound()));
|
|
}
|
|
|
|
nbttagcompound.setTag("ActiveEffects", nbttaglist);
|
|
}
|
|
|
|
}
|
|
|
|
/**+
|
|
* (abstract) Protected helper method to read subclass entity
|
|
* data from NBT.
|
|
*/
|
|
public void readEntityFromNBT(NBTTagCompound nbttagcompound) {
|
|
this.setAbsorptionAmount(nbttagcompound.getFloat("AbsorptionAmount"));
|
|
|
|
if (nbttagcompound.hasKey("ActiveEffects", 9)) {
|
|
NBTTagList nbttaglist = nbttagcompound.getTagList("ActiveEffects", 10);
|
|
|
|
for (int i = 0; i < nbttaglist.tagCount(); ++i) {
|
|
NBTTagCompound nbttagcompound1 = nbttaglist.getCompoundTagAt(i);
|
|
PotionEffect potioneffect = PotionEffect.readCustomPotionEffectFromNBT(nbttagcompound1);
|
|
if (potioneffect != null) {
|
|
this.activePotionsMap.put(Integer.valueOf(potioneffect.getPotionID()), potioneffect);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (nbttagcompound.hasKey("HealF", 99)) {
|
|
this.setHealth(nbttagcompound.getFloat("HealF"));
|
|
} else {
|
|
NBTBase nbtbase = nbttagcompound.getTag("Health");
|
|
if (nbtbase == null) {
|
|
this.setHealth(this.getMaxHealth());
|
|
} else if (nbtbase.getId() == 5) {
|
|
this.setHealth(((NBTTagFloat) nbtbase).getFloat());
|
|
} else if (nbtbase.getId() == 2) {
|
|
this.setHealth((float) ((NBTTagShort) nbtbase).getShort());
|
|
}
|
|
}
|
|
|
|
this.hurtTime = nbttagcompound.getShort("HurtTime");
|
|
this.deathTime = nbttagcompound.getShort("DeathTime");
|
|
this.revengeTimer = nbttagcompound.getInteger("HurtByTimestamp");
|
|
}
|
|
|
|
protected void updatePotionEffects() {
|
|
Iterator iterator = this.activePotionsMap.keySet().iterator();
|
|
|
|
while (iterator.hasNext()) {
|
|
Integer integer = (Integer) iterator.next();
|
|
PotionEffect potioneffect = (PotionEffect) this.activePotionsMap.get(integer);
|
|
if (potioneffect.onUpdate(this) && potioneffect.getDuration() % 600 == 0) {
|
|
this.onChangedPotionEffect(potioneffect, false);
|
|
}
|
|
}
|
|
|
|
this.potionsNeedUpdate = false;
|
|
|
|
int i = this.dataWatcher.getWatchableObjectInt(7);
|
|
boolean flag1 = this.dataWatcher.getWatchableObjectByte(8) > 0;
|
|
if (i > 0) {
|
|
boolean flag = false;
|
|
if (!this.isInvisible()) {
|
|
flag = this.rand.nextBoolean();
|
|
} else {
|
|
flag = this.rand.nextInt(15) == 0;
|
|
}
|
|
|
|
if (flag1) {
|
|
flag &= this.rand.nextInt(5) == 0;
|
|
}
|
|
|
|
if (flag && i > 0) {
|
|
double d0 = (double) (i >> 16 & 255) / 255.0D;
|
|
double d1 = (double) (i >> 8 & 255) / 255.0D;
|
|
double d2 = (double) (i >> 0 & 255) / 255.0D;
|
|
this.worldObj.spawnParticle(flag1 ? EnumParticleTypes.SPELL_MOB_AMBIENT : EnumParticleTypes.SPELL_MOB,
|
|
this.posX + (this.rand.nextDouble() - 0.5D) * (double) this.width,
|
|
this.posY + this.rand.nextDouble() * (double) this.height,
|
|
this.posZ + (this.rand.nextDouble() - 0.5D) * (double) this.width, d0, d1, d2, new int[0]);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
/**+
|
|
* Clears potion metadata values if the entity has no potion
|
|
* effects. Otherwise, updates potion effect color, ambience,
|
|
* and invisibility metadata values
|
|
*/
|
|
protected void updatePotionMetadata() {
|
|
if (this.activePotionsMap.isEmpty()) {
|
|
this.resetPotionEffectMetadata();
|
|
this.setInvisible(false);
|
|
} else {
|
|
int i = PotionHelper.calcPotionLiquidColor(this.activePotionsMap.values());
|
|
this.dataWatcher.updateObject(8,
|
|
Byte.valueOf((byte) (PotionHelper.getAreAmbient(this.activePotionsMap.values()) ? 1 : 0)));
|
|
this.dataWatcher.updateObject(7, Integer.valueOf(i));
|
|
this.setInvisible(this.isPotionActive(Potion.invisibility.id));
|
|
}
|
|
|
|
}
|
|
|
|
/**+
|
|
* Resets the potion effect color and ambience metadata values
|
|
*/
|
|
protected void resetPotionEffectMetadata() {
|
|
this.dataWatcher.updateObject(8, Byte.valueOf((byte) 0));
|
|
this.dataWatcher.updateObject(7, Integer.valueOf(0));
|
|
}
|
|
|
|
public void clearActivePotions() {
|
|
|
|
}
|
|
|
|
public Collection<PotionEffect> getActivePotionEffects() {
|
|
return this.activePotionsMap.values();
|
|
}
|
|
|
|
public boolean isPotionActive(int potionId) {
|
|
return this.activePotionsMap.containsKey(Integer.valueOf(potionId));
|
|
}
|
|
|
|
public boolean isPotionActive(Potion potionIn) {
|
|
return this.activePotionsMap.containsKey(Integer.valueOf(potionIn.id));
|
|
}
|
|
|
|
/**+
|
|
* returns the PotionEffect for the supplied Potion if it is
|
|
* active, null otherwise.
|
|
*/
|
|
public PotionEffect getActivePotionEffect(Potion potionIn) {
|
|
return (PotionEffect) this.activePotionsMap.get(Integer.valueOf(potionIn.id));
|
|
}
|
|
|
|
/**+
|
|
* adds a PotionEffect to the entity
|
|
*/
|
|
public void addPotionEffect(PotionEffect potioneffectIn) {
|
|
if (this.isPotionApplicable(potioneffectIn)) {
|
|
if (this.activePotionsMap.containsKey(Integer.valueOf(potioneffectIn.getPotionID()))) {
|
|
((PotionEffect) this.activePotionsMap.get(Integer.valueOf(potioneffectIn.getPotionID())))
|
|
.combine(potioneffectIn);
|
|
this.onChangedPotionEffect(
|
|
(PotionEffect) this.activePotionsMap.get(Integer.valueOf(potioneffectIn.getPotionID())), true);
|
|
} else {
|
|
this.activePotionsMap.put(Integer.valueOf(potioneffectIn.getPotionID()), potioneffectIn);
|
|
this.onNewPotionEffect(potioneffectIn);
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
public boolean isPotionApplicable(PotionEffect potioneffectIn) {
|
|
if (this.getCreatureAttribute() == EnumCreatureAttribute.UNDEAD) {
|
|
int i = potioneffectIn.getPotionID();
|
|
if (i == Potion.regeneration.id || i == Potion.poison.id) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**+
|
|
* Returns true if this entity is undead.
|
|
*/
|
|
public boolean isEntityUndead() {
|
|
return this.getCreatureAttribute() == EnumCreatureAttribute.UNDEAD;
|
|
}
|
|
|
|
/**+
|
|
* Remove the speified potion effect from this entity.
|
|
*/
|
|
public void removePotionEffectClient(int potionId) {
|
|
this.activePotionsMap.remove(Integer.valueOf(potionId));
|
|
}
|
|
|
|
/**+
|
|
* Remove the specified potion effect from this entity.
|
|
*/
|
|
public void removePotionEffect(int potionId) {
|
|
PotionEffect potioneffect = (PotionEffect) this.activePotionsMap.remove(Integer.valueOf(potionId));
|
|
if (potioneffect != null) {
|
|
this.onFinishedPotionEffect(potioneffect);
|
|
}
|
|
|
|
}
|
|
|
|
protected void onNewPotionEffect(PotionEffect potioneffect) {
|
|
this.potionsNeedUpdate = true;
|
|
}
|
|
|
|
protected void onChangedPotionEffect(PotionEffect potioneffect, boolean flag) {
|
|
this.potionsNeedUpdate = true;
|
|
}
|
|
|
|
protected void onFinishedPotionEffect(PotionEffect potioneffect) {
|
|
this.potionsNeedUpdate = true;
|
|
}
|
|
|
|
/**+
|
|
* Heal living entity (param: amount of half-hearts)
|
|
*/
|
|
public void heal(float f) {
|
|
float f1 = this.getHealth();
|
|
if (f1 > 0.0F) {
|
|
this.setHealth(f1 + f);
|
|
}
|
|
|
|
}
|
|
|
|
public final float getHealth() {
|
|
return this.dataWatcher.getWatchableObjectFloat(6);
|
|
}
|
|
|
|
public void setHealth(float health) {
|
|
this.dataWatcher.updateObject(6, Float.valueOf(MathHelper.clamp_float(health, 0.0F, this.getMaxHealth())));
|
|
}
|
|
|
|
/**+
|
|
* Called when the entity is attacked.
|
|
*/
|
|
public boolean attackEntityFrom(DamageSource damagesource, float f) {
|
|
return false;
|
|
}
|
|
|
|
/**+
|
|
* Renders broken item particles using the given ItemStack
|
|
*/
|
|
public void renderBrokenItemStack(ItemStack stack) {
|
|
this.playSound("random.break", 0.8F, 0.8F + this.worldObj.rand.nextFloat() * 0.4F);
|
|
|
|
for (int i = 0; i < 5; ++i) {
|
|
Vec3 vec3 = new Vec3(((double) this.rand.nextFloat() - 0.5D) * 0.1D, Math.random() * 0.1D + 0.1D, 0.0D);
|
|
vec3 = vec3.rotatePitch(-this.rotationPitch * 3.1415927F / 180.0F);
|
|
vec3 = vec3.rotateYaw(-this.rotationYaw * 3.1415927F / 180.0F);
|
|
double d0 = (double) (-this.rand.nextFloat()) * 0.6D - 0.3D;
|
|
Vec3 vec31 = new Vec3(((double) this.rand.nextFloat() - 0.5D) * 0.3D, d0, 0.6D);
|
|
vec31 = vec31.rotatePitch(-this.rotationPitch * 3.1415927F / 180.0F);
|
|
vec31 = vec31.rotateYaw(-this.rotationYaw * 3.1415927F / 180.0F);
|
|
vec31 = vec31.addVector(this.posX, this.posY + (double) this.getEyeHeight(), this.posZ);
|
|
this.worldObj.spawnParticle(EnumParticleTypes.ITEM_CRACK, vec31.xCoord, vec31.yCoord, vec31.zCoord,
|
|
vec3.xCoord, vec3.yCoord + 0.05D, vec3.zCoord, new int[] { Item.getIdFromItem(stack.getItem()) });
|
|
}
|
|
|
|
}
|
|
|
|
/**+
|
|
* Called when the mob's health reaches 0.
|
|
*/
|
|
public void onDeath(DamageSource damagesource) {
|
|
Entity entity = damagesource.getEntity();
|
|
EntityLivingBase entitylivingbase = this.func_94060_bK();
|
|
if (this.scoreValue >= 0 && entitylivingbase != null) {
|
|
entitylivingbase.addToPlayerScore(this, this.scoreValue);
|
|
}
|
|
|
|
if (entity != null) {
|
|
entity.onKillEntity(this);
|
|
}
|
|
|
|
this.dead = true;
|
|
this.getCombatTracker().reset();
|
|
|
|
this.worldObj.setEntityState(this, (byte) 3);
|
|
}
|
|
|
|
/**+
|
|
* Drop the equipment for this entity.
|
|
*/
|
|
protected void dropEquipment(boolean parFlag, int parInt1) {
|
|
}
|
|
|
|
/**+
|
|
* knocks back this entity
|
|
*/
|
|
public void knockBack(Entity entityIn, float parFloat1, double parDouble1, double parDouble2) {
|
|
if (this.rand.nextDouble() >= this.getEntityAttribute(SharedMonsterAttributes.knockbackResistance)
|
|
.getAttributeValue()) {
|
|
this.isAirBorne = true;
|
|
float f = MathHelper.sqrt_double(parDouble1 * parDouble1 + parDouble2 * parDouble2);
|
|
float f1 = 0.4F;
|
|
this.motionX /= 2.0D;
|
|
this.motionY /= 2.0D;
|
|
this.motionZ /= 2.0D;
|
|
this.motionX -= parDouble1 / (double) f * (double) f1;
|
|
this.motionY += (double) f1;
|
|
this.motionZ -= parDouble2 / (double) f * (double) f1;
|
|
if (this.motionY > 0.4000000059604645D) {
|
|
this.motionY = 0.4000000059604645D;
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
/**+
|
|
* Returns the sound this mob makes when it is hurt.
|
|
*/
|
|
protected String getHurtSound() {
|
|
return "game.neutral.hurt";
|
|
}
|
|
|
|
/**+
|
|
* Returns the sound this mob makes on death.
|
|
*/
|
|
protected String getDeathSound() {
|
|
return "game.neutral.die";
|
|
}
|
|
|
|
/**+
|
|
* Causes this Entity to drop a random item.
|
|
*/
|
|
protected void addRandomDrop() {
|
|
}
|
|
|
|
/**+
|
|
* Drop 0-2 items of this living's type
|
|
*/
|
|
protected void dropFewItems(boolean parFlag, int parInt1) {
|
|
}
|
|
|
|
/**+
|
|
* returns true if this entity is by a ladder, false otherwise
|
|
*/
|
|
public boolean isOnLadder() {
|
|
int i = MathHelper.floor_double(this.posX);
|
|
int j = MathHelper.floor_double(this.getEntityBoundingBox().minY);
|
|
int k = MathHelper.floor_double(this.posZ);
|
|
Block block = this.worldObj.getBlockState(new BlockPos(i, j, k)).getBlock();
|
|
return (block == Blocks.ladder || block == Blocks.vine)
|
|
&& (!(this instanceof EntityPlayer) || !((EntityPlayer) this).isSpectator());
|
|
}
|
|
|
|
/**+
|
|
* Checks whether target entity is alive.
|
|
*/
|
|
public boolean isEntityAlive() {
|
|
return !this.isDead && this.getHealth() > 0.0F;
|
|
}
|
|
|
|
public void fall(float f, float f1) {
|
|
super.fall(f, f1);
|
|
PotionEffect potioneffect = this.getActivePotionEffect(Potion.jump);
|
|
float f2 = potioneffect != null ? (float) (potioneffect.getAmplifier() + 1) : 0.0F;
|
|
int i = MathHelper.ceiling_float_int((f - 3.0F - f2) * f1);
|
|
if (i > 0) {
|
|
this.playSound(this.getFallSoundString(i), 1.0F, 1.0F);
|
|
this.attackEntityFrom(DamageSource.fall, (float) i);
|
|
int j = MathHelper.floor_double(this.posX);
|
|
int k = MathHelper.floor_double(this.posY - 0.20000000298023224D);
|
|
int l = MathHelper.floor_double(this.posZ);
|
|
Block block = this.worldObj.getBlockState(new BlockPos(j, k, l)).getBlock();
|
|
if (block.getMaterial() != Material.air) {
|
|
Block.SoundType block$soundtype = block.stepSound;
|
|
this.playSound(block$soundtype.getStepSound(), block$soundtype.getVolume() * 0.5F,
|
|
block$soundtype.getFrequency() * 0.75F);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
protected String getFallSoundString(int damageValue) {
|
|
return damageValue > 4 ? "game.neutral.hurt.fall.big" : "game.neutral.hurt.fall.small";
|
|
}
|
|
|
|
/**+
|
|
* Setups the entity to do the hurt animation. Only used by
|
|
* packets in multiplayer.
|
|
*/
|
|
public void performHurtAnimation() {
|
|
this.hurtTime = this.maxHurtTime = 10;
|
|
this.attackedAtYaw = 0.0F;
|
|
}
|
|
|
|
/**+
|
|
* Returns the current armor value as determined by a call to
|
|
* InventoryPlayer.getTotalArmorValue
|
|
*/
|
|
public int getTotalArmorValue() {
|
|
int i = 0;
|
|
|
|
for (ItemStack itemstack : this.getInventory()) {
|
|
if (itemstack != null && itemstack.getItem() instanceof ItemArmor) {
|
|
int j = ((ItemArmor) itemstack.getItem()).damageReduceAmount;
|
|
i += j;
|
|
}
|
|
}
|
|
|
|
return i;
|
|
}
|
|
|
|
protected void damageArmor(float parFloat1) {
|
|
}
|
|
|
|
/**+
|
|
* Reduces damage, depending on armor
|
|
*/
|
|
protected float applyArmorCalculations(DamageSource source, float damage) {
|
|
if (!source.isUnblockable()) {
|
|
int i = 25 - this.getTotalArmorValue();
|
|
float f = damage * (float) i;
|
|
this.damageArmor(damage);
|
|
damage = f / 25.0F;
|
|
}
|
|
|
|
return damage;
|
|
}
|
|
|
|
/**+
|
|
* Reduces damage, depending on potions
|
|
*/
|
|
protected float applyPotionDamageCalculations(DamageSource source, float damage) {
|
|
if (source.isDamageAbsolute()) {
|
|
return damage;
|
|
} else {
|
|
if (this.isPotionActive(Potion.resistance) && source != DamageSource.outOfWorld) {
|
|
int i = (this.getActivePotionEffect(Potion.resistance).getAmplifier() + 1) * 5;
|
|
int j = 25 - i;
|
|
float f = damage * (float) j;
|
|
damage = f / 25.0F;
|
|
}
|
|
|
|
if (damage <= 0.0F) {
|
|
return 0.0F;
|
|
} else {
|
|
int k = EnchantmentHelper.getEnchantmentModifierDamage(this.getInventory(), source);
|
|
if (k > 20) {
|
|
k = 20;
|
|
}
|
|
|
|
if (k > 0 && k <= 20) {
|
|
int l = 25 - k;
|
|
float f1 = damage * (float) l;
|
|
damage = f1 / 25.0F;
|
|
}
|
|
|
|
return damage;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**+
|
|
* Deals damage to the entity. If its a EntityPlayer then will
|
|
* take damage from the armor first and then health second with
|
|
* the reduced value. Args: damageAmount
|
|
*/
|
|
protected void damageEntity(DamageSource damagesource, float f) {
|
|
if (!this.isEntityInvulnerable(damagesource)) {
|
|
f = this.applyArmorCalculations(damagesource, f);
|
|
f = this.applyPotionDamageCalculations(damagesource, f);
|
|
float f1 = f;
|
|
f = Math.max(f - this.getAbsorptionAmount(), 0.0F);
|
|
this.setAbsorptionAmount(this.getAbsorptionAmount() - (f1 - f));
|
|
if (f != 0.0F) {
|
|
float f2 = this.getHealth();
|
|
this.setHealth(f2 - f);
|
|
this.getCombatTracker().trackDamage(damagesource, f2, f);
|
|
this.setAbsorptionAmount(this.getAbsorptionAmount() - f);
|
|
}
|
|
}
|
|
}
|
|
|
|
public CombatTracker getCombatTracker() {
|
|
return this._combatTracker;
|
|
}
|
|
|
|
public EntityLivingBase func_94060_bK() {
|
|
return (EntityLivingBase) (this._combatTracker.func_94550_c() != null ? this._combatTracker.func_94550_c()
|
|
: (this.attackingPlayer != null ? this.attackingPlayer
|
|
: (this.entityLivingToAttack != null ? this.entityLivingToAttack : null)));
|
|
}
|
|
|
|
public final float getMaxHealth() {
|
|
return (float) this.getEntityAttribute(SharedMonsterAttributes.maxHealth).getAttributeValue();
|
|
}
|
|
|
|
/**+
|
|
* counts the amount of arrows stuck in the entity. getting hit
|
|
* by arrows increases this, used in rendering
|
|
*/
|
|
public final int getArrowCountInEntity() {
|
|
return this.dataWatcher.getWatchableObjectByte(9);
|
|
}
|
|
|
|
/**+
|
|
* sets the amount of arrows stuck in the entity. used for
|
|
* rendering those
|
|
*/
|
|
public final void setArrowCountInEntity(int count) {
|
|
this.dataWatcher.updateObject(9, Byte.valueOf((byte) count));
|
|
}
|
|
|
|
/**+
|
|
* Returns an integer indicating the end point of the swing
|
|
* animation, used by {@link #swingProgress} to provide a
|
|
* progress indicator. Takes dig speed enchantments into
|
|
* account.
|
|
*/
|
|
private int getArmSwingAnimationEnd() {
|
|
return this.isPotionActive(Potion.digSpeed)
|
|
? 6 - (1 + this.getActivePotionEffect(Potion.digSpeed).getAmplifier()) * 1
|
|
: (this.isPotionActive(Potion.digSlowdown)
|
|
? 6 + (1 + this.getActivePotionEffect(Potion.digSlowdown).getAmplifier()) * 2
|
|
: 6);
|
|
}
|
|
|
|
/**+
|
|
* Swings the item the player is holding.
|
|
*/
|
|
public void swingItem() {
|
|
if (!this.isSwingInProgress || this.swingProgressInt >= this.getArmSwingAnimationEnd() / 2
|
|
|| this.swingProgressInt < 0) {
|
|
this.swingProgressInt = -1;
|
|
this.isSwingInProgress = true;
|
|
}
|
|
|
|
}
|
|
|
|
public void handleStatusUpdate(byte b0) {
|
|
if (b0 == 2) {
|
|
this.limbSwingAmount = 1.5F;
|
|
this.hurtResistantTime = this.maxHurtResistantTime;
|
|
this.hurtTime = this.maxHurtTime = 10;
|
|
this.attackedAtYaw = 0.0F;
|
|
String s = this.getHurtSound();
|
|
if (s != null) {
|
|
this.playSound(this.getHurtSound(), this.getSoundVolume(),
|
|
(this.rand.nextFloat() - this.rand.nextFloat()) * 0.2F + 1.0F);
|
|
}
|
|
|
|
this.attackEntityFrom(DamageSource.generic, 0.0F);
|
|
} else if (b0 == 3) {
|
|
String s1 = this.getDeathSound();
|
|
if (s1 != null) {
|
|
this.playSound(this.getDeathSound(), this.getSoundVolume(),
|
|
(this.rand.nextFloat() - this.rand.nextFloat()) * 0.2F + 1.0F);
|
|
}
|
|
|
|
this.setHealth(0.0F);
|
|
this.onDeath(DamageSource.generic);
|
|
} else {
|
|
super.handleStatusUpdate(b0);
|
|
}
|
|
|
|
}
|
|
|
|
/**+
|
|
* sets the dead flag. Used when you fall off the bottom of the
|
|
* world.
|
|
*/
|
|
protected void kill() {
|
|
this.attackEntityFrom(DamageSource.outOfWorld, 4.0F);
|
|
}
|
|
|
|
/**+
|
|
* Updates the arm swing progress counters and animation
|
|
* progress
|
|
*/
|
|
protected void updateArmSwingProgress() {
|
|
int i = this.getArmSwingAnimationEnd();
|
|
if (this.isSwingInProgress) {
|
|
++this.swingProgressInt;
|
|
if (this.swingProgressInt >= i) {
|
|
this.swingProgressInt = 0;
|
|
this.isSwingInProgress = false;
|
|
}
|
|
} else {
|
|
this.swingProgressInt = 0;
|
|
}
|
|
|
|
this.swingProgress = (float) this.swingProgressInt / (float) i;
|
|
}
|
|
|
|
public IAttributeInstance getEntityAttribute(IAttribute attribute) {
|
|
return this.getAttributeMap().getAttributeInstance(attribute);
|
|
}
|
|
|
|
public BaseAttributeMap getAttributeMap() {
|
|
if (this.attributeMap == null) {
|
|
this.attributeMap = new ServersideAttributeMap();
|
|
}
|
|
return this.attributeMap;
|
|
}
|
|
|
|
/**+
|
|
* Get this Entity's EnumCreatureAttribute
|
|
*/
|
|
public EnumCreatureAttribute getCreatureAttribute() {
|
|
return EnumCreatureAttribute.UNDEFINED;
|
|
}
|
|
|
|
public abstract ItemStack getHeldItem();
|
|
|
|
public abstract ItemStack getEquipmentInSlot(int var1);
|
|
|
|
public abstract ItemStack getCurrentArmor(int var1);
|
|
|
|
public abstract void setCurrentItemOrArmor(int var1, ItemStack var2);
|
|
|
|
/**+
|
|
* Set sprinting switch for Entity.
|
|
*/
|
|
public void setSprinting(boolean flag) {
|
|
super.setSprinting(flag);
|
|
IAttributeInstance iattributeinstance = this.getEntityAttribute(SharedMonsterAttributes.movementSpeed);
|
|
if (iattributeinstance.getModifier(sprintingSpeedBoostModifierUUID) != null) {
|
|
iattributeinstance.removeModifier(sprintingSpeedBoostModifier);
|
|
}
|
|
|
|
if (flag) {
|
|
iattributeinstance.applyModifier(sprintingSpeedBoostModifier);
|
|
}
|
|
|
|
}
|
|
|
|
public abstract ItemStack[] getInventory();
|
|
|
|
/**+
|
|
* Returns the volume for the sounds this mob makes.
|
|
*/
|
|
protected float getSoundVolume() {
|
|
return 1.0F;
|
|
}
|
|
|
|
/**+
|
|
* Gets the pitch of living sounds in living entities.
|
|
*/
|
|
protected float getSoundPitch() {
|
|
return this.isChild() ? (this.rand.nextFloat() - this.rand.nextFloat()) * 0.2F + 1.5F
|
|
: (this.rand.nextFloat() - this.rand.nextFloat()) * 0.2F + 1.0F;
|
|
}
|
|
|
|
/**+
|
|
* Dead and sleeping entities cannot move
|
|
*/
|
|
protected boolean isMovementBlocked() {
|
|
return this.getHealth() <= 0.0F;
|
|
}
|
|
|
|
/**+
|
|
* Moves the entity to a position out of the way of its mount.
|
|
*/
|
|
public void dismountEntity(Entity parEntity) {
|
|
double d0 = parEntity.posX;
|
|
double d1 = parEntity.getEntityBoundingBox().minY + (double) parEntity.height;
|
|
double d2 = parEntity.posZ;
|
|
byte b0 = 1;
|
|
|
|
for (int i = -b0; i <= b0; ++i) {
|
|
for (int j = -b0; j < b0; ++j) {
|
|
if (i != 0 || j != 0) {
|
|
int k = (int) (this.posX + (double) i);
|
|
int l = (int) (this.posZ + (double) j);
|
|
AxisAlignedBB axisalignedbb = this.getEntityBoundingBox().offset((double) i, 1.0D, (double) j);
|
|
if (this.worldObj.func_147461_a(axisalignedbb).isEmpty()) {
|
|
if (World.doesBlockHaveSolidTopSurface(this.worldObj, new BlockPos(k, (int) this.posY, l))) {
|
|
this.setPositionAndUpdate(this.posX + (double) i, this.posY + 1.0D, this.posZ + (double) j);
|
|
return;
|
|
}
|
|
|
|
if (World.doesBlockHaveSolidTopSurface(this.worldObj, new BlockPos(k, (int) this.posY - 1, l))
|
|
|| this.worldObj.getBlockState(new BlockPos(k, (int) this.posY - 1, l)).getBlock()
|
|
.getMaterial() == Material.water) {
|
|
d0 = this.posX + (double) i;
|
|
d1 = this.posY + 1.0D;
|
|
d2 = this.posZ + (double) j;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
this.setPositionAndUpdate(d0, d1, d2);
|
|
}
|
|
|
|
public boolean getAlwaysRenderNameTagForRender() {
|
|
return false;
|
|
}
|
|
|
|
protected float getJumpUpwardsMotion() {
|
|
return 0.42F;
|
|
}
|
|
|
|
/**+
|
|
* Causes this entity to do an upwards motion (jumping).
|
|
*/
|
|
protected void jump() {
|
|
this.motionY = (double) this.getJumpUpwardsMotion();
|
|
if (this.isPotionActive(Potion.jump)) {
|
|
this.motionY += (double) ((float) (this.getActivePotionEffect(Potion.jump).getAmplifier() + 1) * 0.1F);
|
|
}
|
|
|
|
if (this.isSprinting()) {
|
|
float f = this.rotationYaw * 0.017453292F;
|
|
this.motionX -= (double) (MathHelper.sin(f) * 0.2F);
|
|
this.motionZ += (double) (MathHelper.cos(f) * 0.2F);
|
|
}
|
|
|
|
this.isAirBorne = true;
|
|
}
|
|
|
|
/**+
|
|
* main AI tick function, replaces updateEntityActionState
|
|
*/
|
|
protected void updateAITick() {
|
|
this.motionY += 0.03999999910593033D;
|
|
}
|
|
|
|
protected void handleJumpLava() {
|
|
this.motionY += 0.03999999910593033D;
|
|
}
|
|
|
|
/**+
|
|
* Moves the entity based on the specified heading. Args:
|
|
* strafe, forward
|
|
*/
|
|
public void moveEntityWithHeading(float f, float f1) {
|
|
if (this.isServerWorld()) {
|
|
if (!this.isInWater() || this instanceof EntityPlayer && ((EntityPlayer) this).capabilities.isFlying) {
|
|
if (!this.isInLava() || this instanceof EntityPlayer && ((EntityPlayer) this).capabilities.isFlying) {
|
|
float f6 = 0.91F;
|
|
if (this.onGround) {
|
|
f6 = this.worldObj.getBlockState(new BlockPos(MathHelper.floor_double(this.posX),
|
|
MathHelper.floor_double(this.getEntityBoundingBox().minY) - 1,
|
|
MathHelper.floor_double(this.posZ))).getBlock().slipperiness * 0.91F;
|
|
}
|
|
|
|
float f2 = 0.16277136F / (f6 * f6 * f6);
|
|
float f7;
|
|
if (this.onGround) {
|
|
f7 = this.getAIMoveSpeed() * f2;
|
|
} else {
|
|
f7 = this.jumpMovementFactor;
|
|
}
|
|
|
|
this.moveFlying(f, f1, f7);
|
|
f6 = 0.91F;
|
|
if (this.onGround) {
|
|
f6 = this.worldObj.getBlockState(new BlockPos(MathHelper.floor_double(this.posX),
|
|
MathHelper.floor_double(this.getEntityBoundingBox().minY) - 1,
|
|
MathHelper.floor_double(this.posZ))).getBlock().slipperiness * 0.91F;
|
|
}
|
|
|
|
if (this.isOnLadder()) {
|
|
float f8 = 0.15F;
|
|
this.motionX = MathHelper.clamp_double(this.motionX, (double) (-f8), (double) f8);
|
|
this.motionZ = MathHelper.clamp_double(this.motionZ, (double) (-f8), (double) f8);
|
|
this.fallDistance = 0.0F;
|
|
if (this.motionY < -0.15D) {
|
|
this.motionY = -0.15D;
|
|
}
|
|
|
|
boolean flag = this.isSneaking() && this instanceof EntityPlayer;
|
|
if (flag && this.motionY < 0.0D) {
|
|
this.motionY = 0.0D;
|
|
}
|
|
}
|
|
|
|
this.moveEntity(this.motionX, this.motionY, this.motionZ);
|
|
if (this.isCollidedHorizontally && this.isOnLadder()) {
|
|
this.motionY = 0.2D;
|
|
}
|
|
|
|
if (!this.worldObj.isBlockLoaded(new BlockPos((int) this.posX, 0, (int) this.posZ))
|
|
|| !this.worldObj.getChunkFromBlockCoords(new BlockPos((int) this.posX, 0, (int) this.posZ))
|
|
.isLoaded()) {
|
|
if (this.posY > 0.0D) {
|
|
this.motionY = -0.1D;
|
|
} else {
|
|
this.motionY = 0.0D;
|
|
}
|
|
} else {
|
|
this.motionY -= 0.08D;
|
|
}
|
|
|
|
this.motionY *= 0.9800000190734863D;
|
|
this.motionX *= (double) f6;
|
|
this.motionZ *= (double) f6;
|
|
} else {
|
|
double d1 = this.posY;
|
|
this.moveFlying(f, f1, 0.02F);
|
|
this.moveEntity(this.motionX, this.motionY, this.motionZ);
|
|
this.motionX *= 0.5D;
|
|
this.motionY *= 0.5D;
|
|
this.motionZ *= 0.5D;
|
|
this.motionY -= 0.02D;
|
|
if (this.isCollidedHorizontally && this.isOffsetPositionInLiquid(this.motionX,
|
|
this.motionY + 0.6000000238418579D - this.posY + d1, this.motionZ)) {
|
|
this.motionY = 0.30000001192092896D;
|
|
}
|
|
}
|
|
} else {
|
|
double d0 = this.posY;
|
|
float f3 = 0.8F;
|
|
float f4 = 0.02F;
|
|
float f5 = (float) EnchantmentHelper.getDepthStriderModifier(this);
|
|
if (f5 > 3.0F) {
|
|
f5 = 3.0F;
|
|
}
|
|
|
|
if (!this.onGround) {
|
|
f5 *= 0.5F;
|
|
}
|
|
|
|
if (f5 > 0.0F) {
|
|
f3 += (0.54600006F - f3) * f5 / 3.0F;
|
|
f4 += (this.getAIMoveSpeed() * 1.0F - f4) * f5 / 3.0F;
|
|
}
|
|
|
|
this.moveFlying(f, f1, f4);
|
|
this.moveEntity(this.motionX, this.motionY, this.motionZ);
|
|
this.motionX *= (double) f3;
|
|
this.motionY *= 0.800000011920929D;
|
|
this.motionZ *= (double) f3;
|
|
this.motionY -= 0.02D;
|
|
if (this.isCollidedHorizontally && this.isOffsetPositionInLiquid(this.motionX,
|
|
this.motionY + 0.6000000238418579D - this.posY + d0, this.motionZ)) {
|
|
this.motionY = 0.30000001192092896D;
|
|
}
|
|
}
|
|
}
|
|
|
|
this.prevLimbSwingAmount = this.limbSwingAmount;
|
|
double d2 = this.posX - this.prevPosX;
|
|
double d3 = this.posZ - this.prevPosZ;
|
|
float f9 = MathHelper.sqrt_double(d2 * d2 + d3 * d3) * 4.0F;
|
|
if (f9 > 1.0F) {
|
|
f9 = 1.0F;
|
|
}
|
|
|
|
this.limbSwingAmount += (f9 - this.limbSwingAmount) * 0.4F;
|
|
this.limbSwing += this.limbSwingAmount;
|
|
}
|
|
|
|
/**+
|
|
* the movespeed used for the new AI system
|
|
*/
|
|
public float getAIMoveSpeed() {
|
|
return this.landMovementFactor;
|
|
}
|
|
|
|
/**+
|
|
* set the movespeed used for the new AI system
|
|
*/
|
|
public void setAIMoveSpeed(float speedIn) {
|
|
this.landMovementFactor = speedIn;
|
|
}
|
|
|
|
public boolean attackEntityAsMob(Entity entityIn) {
|
|
this.setLastAttacker(entityIn);
|
|
return false;
|
|
}
|
|
|
|
/**+
|
|
* Returns whether player is sleeping or not
|
|
*/
|
|
public boolean isPlayerSleeping() {
|
|
return false;
|
|
}
|
|
|
|
/**+
|
|
* Called to update the entity's position/logic.
|
|
*/
|
|
public void onUpdate() {
|
|
super.onUpdate();
|
|
this.onLivingUpdate();
|
|
double d0 = this.posX - this.prevPosX;
|
|
double d1 = this.posZ - this.prevPosZ;
|
|
float f = (float) (d0 * d0 + d1 * d1);
|
|
float f1 = this.renderYawOffset;
|
|
float f2 = 0.0F;
|
|
this.prevOnGroundSpeedFactor = this.onGroundSpeedFactor;
|
|
float f3 = 0.0F;
|
|
if (f > 0.0025000002F) {
|
|
f3 = 1.0F;
|
|
f2 = (float) Math.sqrt((double) f) * 3.0F;
|
|
f1 = (float) MathHelper.func_181159_b(d1, d0) * 180.0F / 3.1415927F - 90.0F;
|
|
}
|
|
|
|
if (this.swingProgress > 0.0F) {
|
|
f1 = this.rotationYaw;
|
|
}
|
|
|
|
if (!this.onGround) {
|
|
f3 = 0.0F;
|
|
}
|
|
|
|
this.onGroundSpeedFactor += (f3 - this.onGroundSpeedFactor) * 0.3F;
|
|
this.worldObj.theProfiler.startSection("headTurn");
|
|
f2 = this.func_110146_f(f1, f2);
|
|
this.worldObj.theProfiler.endSection();
|
|
this.worldObj.theProfiler.startSection("rangeChecks");
|
|
|
|
while (this.rotationYaw - this.prevRotationYaw < -180.0F) {
|
|
this.prevRotationYaw -= 360.0F;
|
|
}
|
|
|
|
while (this.rotationYaw - this.prevRotationYaw >= 180.0F) {
|
|
this.prevRotationYaw += 360.0F;
|
|
}
|
|
|
|
while (this.renderYawOffset - this.prevRenderYawOffset < -180.0F) {
|
|
this.prevRenderYawOffset -= 360.0F;
|
|
}
|
|
|
|
while (this.renderYawOffset - this.prevRenderYawOffset >= 180.0F) {
|
|
this.prevRenderYawOffset += 360.0F;
|
|
}
|
|
|
|
while (this.rotationPitch - this.prevRotationPitch < -180.0F) {
|
|
this.prevRotationPitch -= 360.0F;
|
|
}
|
|
|
|
while (this.rotationPitch - this.prevRotationPitch >= 180.0F) {
|
|
this.prevRotationPitch += 360.0F;
|
|
}
|
|
|
|
while (this.rotationYawHead - this.prevRotationYawHead < -180.0F) {
|
|
this.prevRotationYawHead -= 360.0F;
|
|
}
|
|
|
|
while (this.rotationYawHead - this.prevRotationYawHead >= 180.0F) {
|
|
this.prevRotationYawHead += 360.0F;
|
|
}
|
|
|
|
this.worldObj.theProfiler.endSection();
|
|
this.movedDistance += f2;
|
|
}
|
|
|
|
protected float func_110146_f(float parFloat1, float parFloat2) {
|
|
float f = MathHelper.wrapAngleTo180_float(parFloat1 - this.renderYawOffset);
|
|
this.renderYawOffset += f * 0.3F;
|
|
float f1 = MathHelper.wrapAngleTo180_float(this.rotationYaw - this.renderYawOffset);
|
|
boolean flag = f1 < -90.0F || f1 >= 90.0F;
|
|
if (f1 < -75.0F) {
|
|
f1 = -75.0F;
|
|
}
|
|
|
|
if (f1 >= 75.0F) {
|
|
f1 = 75.0F;
|
|
}
|
|
|
|
this.renderYawOffset = this.rotationYaw - f1;
|
|
if (f1 * f1 > 2500.0F) {
|
|
this.renderYawOffset += f1 * 0.2F;
|
|
}
|
|
|
|
if (flag) {
|
|
parFloat2 *= -1.0F;
|
|
}
|
|
|
|
return parFloat2;
|
|
}
|
|
|
|
/**+
|
|
* Called frequently so the entity can update its state every
|
|
* tick as required. For example, zombies and skeletons use this
|
|
* to react to sunlight and start to burn.
|
|
*/
|
|
public void onLivingUpdate() {
|
|
if (this.jumpTicks > 0) {
|
|
--this.jumpTicks;
|
|
}
|
|
|
|
if (this.newPosRotationIncrements > 0) {
|
|
double d0 = this.posX + (this.newPosX - this.posX) / (double) this.newPosRotationIncrements;
|
|
double d1 = this.posY + (this.newPosY - this.posY) / (double) this.newPosRotationIncrements;
|
|
double d2 = this.posZ + (this.newPosZ - this.posZ) / (double) this.newPosRotationIncrements;
|
|
double d3 = MathHelper.wrapAngleTo180_double(this.newRotationYaw - (double) this.rotationYaw);
|
|
this.rotationYaw = (float) ((double) this.rotationYaw + d3 / (double) this.newPosRotationIncrements);
|
|
this.rotationPitch = (float) ((double) this.rotationPitch
|
|
+ (this.newRotationPitch - (double) this.rotationPitch) / (double) this.newPosRotationIncrements);
|
|
--this.newPosRotationIncrements;
|
|
this.setPosition(d0, d1, d2);
|
|
this.setRotation(this.rotationYaw, this.rotationPitch);
|
|
} else if (!this.isServerWorld()) {
|
|
this.motionX *= 0.98D;
|
|
this.motionY *= 0.98D;
|
|
this.motionZ *= 0.98D;
|
|
}
|
|
|
|
if (Math.abs(this.motionX) < 0.005D) {
|
|
this.motionX = 0.0D;
|
|
}
|
|
|
|
if (Math.abs(this.motionY) < 0.005D) {
|
|
this.motionY = 0.0D;
|
|
}
|
|
|
|
if (Math.abs(this.motionZ) < 0.005D) {
|
|
this.motionZ = 0.0D;
|
|
}
|
|
|
|
this.worldObj.theProfiler.startSection("ai");
|
|
if (this.isMovementBlocked()) {
|
|
this.isJumping = false;
|
|
this.moveStrafing = 0.0F;
|
|
this.moveForward = 0.0F;
|
|
this.randomYawVelocity = 0.0F;
|
|
} else if (this.isServerWorld()) {
|
|
this.worldObj.theProfiler.startSection("newAi");
|
|
this.updateEntityActionState();
|
|
this.worldObj.theProfiler.endSection();
|
|
}
|
|
|
|
this.worldObj.theProfiler.endSection();
|
|
this.worldObj.theProfiler.startSection("jump");
|
|
if (this.isJumping) {
|
|
if (this.isInWater()) {
|
|
this.updateAITick();
|
|
} else if (this.isInLava()) {
|
|
this.handleJumpLava();
|
|
} else if (this.onGround && this.jumpTicks == 0) {
|
|
this.jump();
|
|
this.jumpTicks = 10;
|
|
}
|
|
} else {
|
|
this.jumpTicks = 0;
|
|
}
|
|
|
|
this.worldObj.theProfiler.endSection();
|
|
this.worldObj.theProfiler.startSection("travel");
|
|
this.moveStrafing *= 0.98F;
|
|
this.moveForward *= 0.98F;
|
|
this.randomYawVelocity *= 0.9F;
|
|
this.moveEntityWithHeading(this.moveStrafing, this.moveForward);
|
|
this.worldObj.theProfiler.endSection();
|
|
this.worldObj.theProfiler.startSection("push");
|
|
this.worldObj.theProfiler.endSection();
|
|
}
|
|
|
|
protected void updateEntityActionState() {
|
|
}
|
|
|
|
protected void collideWithNearbyEntities() {
|
|
List list = this.worldObj.getEntitiesInAABBexcluding(this,
|
|
this.getEntityBoundingBox().expand(0.20000000298023224D, 0.0D, 0.20000000298023224D),
|
|
Predicates.and(EntitySelectors.NOT_SPECTATING, new Predicate<Entity>() {
|
|
public boolean apply(Entity entity1) {
|
|
return entity1.canBePushed();
|
|
}
|
|
}));
|
|
if (!list.isEmpty()) {
|
|
for (int i = 0; i < list.size(); ++i) {
|
|
Entity entity = (Entity) list.get(i);
|
|
this.collideWithEntity(entity);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
protected void collideWithEntity(Entity parEntity) {
|
|
parEntity.applyEntityCollision(this);
|
|
}
|
|
|
|
/**+
|
|
* Called when a player mounts an entity. e.g. mounts a pig,
|
|
* mounts a boat.
|
|
*/
|
|
public void mountEntity(Entity entity) {
|
|
if (this.ridingEntity != null && entity == null) {
|
|
if (this.ridingEntity != null) {
|
|
this.ridingEntity.riddenByEntity = null;
|
|
}
|
|
|
|
this.ridingEntity = null;
|
|
} else {
|
|
super.mountEntity(entity);
|
|
}
|
|
}
|
|
|
|
/**+
|
|
* Handles updating while being ridden by an entity
|
|
*/
|
|
public void updateRidden() {
|
|
super.updateRidden();
|
|
this.prevOnGroundSpeedFactor = this.onGroundSpeedFactor;
|
|
this.onGroundSpeedFactor = 0.0F;
|
|
this.fallDistance = 0.0F;
|
|
}
|
|
|
|
public void setPositionAndRotation2(double d0, double d1, double d2, float f, float f1, int i, boolean var10) {
|
|
this.newPosX = d0;
|
|
this.newPosY = d1;
|
|
this.newPosZ = d2;
|
|
this.newRotationYaw = (double) f;
|
|
this.newRotationPitch = (double) f1;
|
|
this.newPosRotationIncrements = i;
|
|
}
|
|
|
|
public void setJumping(boolean parFlag) {
|
|
this.isJumping = parFlag;
|
|
}
|
|
|
|
/**+
|
|
* Called whenever an item is picked up from walking over it.
|
|
* Args: pickedUpEntity, stackSize
|
|
*/
|
|
public void onItemPickup(Entity entity, int var2) {
|
|
|
|
}
|
|
|
|
/**+
|
|
* returns true if the entity provided in the argument can be
|
|
* seen. (Raytrace)
|
|
*/
|
|
public boolean canEntityBeSeen(Entity entityIn) {
|
|
return this.worldObj.rayTraceBlocks(new Vec3(this.posX, this.posY + (double) this.getEyeHeight(), this.posZ),
|
|
new Vec3(entityIn.posX, entityIn.posY + (double) entityIn.getEyeHeight(), entityIn.posZ)) == null;
|
|
}
|
|
|
|
/**+
|
|
* returns a (normalized) vector of where this entity is looking
|
|
*/
|
|
public Vec3 getLookVec() {
|
|
return this.getLook(1.0F);
|
|
}
|
|
|
|
/**+
|
|
* interpolated look vector
|
|
*/
|
|
public Vec3 getLook(float f) {
|
|
if (f == 1.0F) {
|
|
return this.getVectorForRotation(this.rotationPitch, this.rotationYawHead);
|
|
} else {
|
|
float f1 = this.prevRotationPitch + (this.rotationPitch - this.prevRotationPitch) * f;
|
|
float f2 = this.prevRotationYawHead + (this.rotationYawHead - this.prevRotationYawHead) * f;
|
|
return this.getVectorForRotation(f1, f2);
|
|
}
|
|
}
|
|
|
|
/**+
|
|
* Returns where in the swing animation the living entity is
|
|
* (from 0 to 1). Args: partialTickTime
|
|
*/
|
|
public float getSwingProgress(float partialTickTime) {
|
|
float f = this.swingProgress - this.prevSwingProgress;
|
|
if (f < 0.0F) {
|
|
++f;
|
|
}
|
|
|
|
return this.prevSwingProgress + f * partialTickTime;
|
|
}
|
|
|
|
/**+
|
|
* Returns whether the entity is in a server world
|
|
*/
|
|
public boolean isServerWorld() {
|
|
return false;
|
|
}
|
|
|
|
/**+
|
|
* Returns true if other Entities should be prevented from
|
|
* moving through this Entity.
|
|
*/
|
|
public boolean canBeCollidedWith() {
|
|
return !this.isDead;
|
|
}
|
|
|
|
/**+
|
|
* Returns true if this entity should push and be pushed by
|
|
* other entities when colliding.
|
|
*/
|
|
public boolean canBePushed() {
|
|
return !this.isDead;
|
|
}
|
|
|
|
/**+
|
|
* Sets that this entity has been attacked.
|
|
*/
|
|
protected void setBeenAttacked() {
|
|
this.velocityChanged = this.rand.nextDouble() >= this
|
|
.getEntityAttribute(SharedMonsterAttributes.knockbackResistance).getAttributeValue();
|
|
}
|
|
|
|
public float getRotationYawHead() {
|
|
return this.rotationYawHead;
|
|
}
|
|
|
|
/**+
|
|
* Sets the head's yaw rotation of the entity.
|
|
*/
|
|
public void setRotationYawHead(float f) {
|
|
this.rotationYawHead = f;
|
|
}
|
|
|
|
public void func_181013_g(float f) {
|
|
this.renderYawOffset = f;
|
|
}
|
|
|
|
public float getAbsorptionAmount() {
|
|
return this.absorptionAmount;
|
|
}
|
|
|
|
public void setAbsorptionAmount(float amount) {
|
|
if (amount < 0.0F) {
|
|
amount = 0.0F;
|
|
}
|
|
|
|
this.absorptionAmount = amount;
|
|
}
|
|
|
|
public Team getTeam() {
|
|
return this.worldObj.getScoreboard().getPlayersTeam(this.getUniqueID().toString());
|
|
}
|
|
|
|
public boolean isOnSameTeam(EntityLivingBase otherEntity) {
|
|
return this.isOnTeam(otherEntity.getTeam());
|
|
}
|
|
|
|
/**+
|
|
* Returns true if the entity is on a specific team.
|
|
*/
|
|
public boolean isOnTeam(Team parTeam) {
|
|
return this.getTeam() != null ? this.getTeam().isSameTeam(parTeam) : false;
|
|
}
|
|
|
|
/**+
|
|
* Sends an ENTER_COMBAT packet to the client
|
|
*/
|
|
public void sendEnterCombat() {
|
|
}
|
|
|
|
/**+
|
|
* Sends an END_COMBAT packet to the client
|
|
*/
|
|
public void sendEndCombat() {
|
|
}
|
|
|
|
protected void markPotionsDirty() {
|
|
this.potionsNeedUpdate = true;
|
|
}
|
|
} |