mirror of
https://github.com/PaperMC/Paper.git
synced 2025-01-08 03:22:19 +01:00
1809 lines
90 KiB
Diff
1809 lines
90 KiB
Diff
--- a/net/minecraft/world/entity/LivingEntity.java
|
|
+++ b/net/minecraft/world/entity/LivingEntity.java
|
|
@@ -42,6 +_,8 @@
|
|
import net.minecraft.core.particles.ParticleOptions;
|
|
import net.minecraft.core.particles.ParticleTypes;
|
|
import net.minecraft.nbt.CompoundTag;
|
|
+import net.minecraft.nbt.FloatTag;
|
|
+import net.minecraft.nbt.IntTag;
|
|
import net.minecraft.nbt.ListTag;
|
|
import net.minecraft.nbt.NbtOps;
|
|
import net.minecraft.nbt.Tag;
|
|
@@ -136,6 +_,29 @@
|
|
import net.minecraft.world.scores.Scoreboard;
|
|
import org.slf4j.Logger;
|
|
|
|
+// CraftBukkit start
|
|
+import java.util.ArrayList;
|
|
+import java.util.HashSet;
|
|
+import java.util.Set;
|
|
+import java.util.LinkedList;
|
|
+import java.util.UUID;
|
|
+import net.minecraft.world.item.component.Consumable;
|
|
+import org.bukkit.Location;
|
|
+import org.bukkit.craftbukkit.attribute.CraftAttributeMap;
|
|
+import org.bukkit.craftbukkit.event.CraftEventFactory;
|
|
+import org.bukkit.craftbukkit.inventory.CraftItemStack;
|
|
+import org.bukkit.event.entity.ArrowBodyCountChangeEvent;
|
|
+import org.bukkit.event.entity.EntityDamageEvent;
|
|
+import org.bukkit.event.entity.EntityDamageEvent.DamageModifier;
|
|
+import org.bukkit.event.entity.EntityKnockbackEvent;
|
|
+import org.bukkit.event.entity.EntityPotionEffectEvent;
|
|
+import org.bukkit.event.entity.EntityRegainHealthEvent;
|
|
+import org.bukkit.event.entity.EntityRemoveEvent;
|
|
+import org.bukkit.event.entity.EntityResurrectEvent;
|
|
+import org.bukkit.event.entity.EntityTeleportEvent;
|
|
+import org.bukkit.event.player.PlayerItemConsumeEvent;
|
|
+// CraftBukkit end
|
|
+
|
|
public abstract class LivingEntity extends Entity implements Attackable {
|
|
private static final Logger LOGGER = LogUtils.getLogger();
|
|
private static final String TAG_ACTIVE_EFFECTS = "active_effects";
|
|
@@ -266,11 +_,29 @@
|
|
EquipmentSlot.class
|
|
);
|
|
protected float appliedScale = 1.0F;
|
|
+ // CraftBukkit start
|
|
+ public int expToDrop;
|
|
+ public ArrayList<DefaultDrop> drops = new ArrayList<>(); // Paper - Restore vanilla drops behavior
|
|
+ public final org.bukkit.craftbukkit.attribute.CraftAttributeMap craftAttributes;
|
|
+ public boolean collides = true;
|
|
+ public Set<UUID> collidableExemptions = new HashSet<>();
|
|
+ public boolean bukkitPickUpLoot;
|
|
+ public org.bukkit.craftbukkit.entity.CraftLivingEntity getBukkitLivingEntity() { return (org.bukkit.craftbukkit.entity.CraftLivingEntity) super.getBukkitEntity(); } // Paper
|
|
+ public boolean silentDeath = false; // Paper - mark entity as dying silently for cancellable death event
|
|
+ public net.kyori.adventure.util.TriState frictionState = net.kyori.adventure.util.TriState.NOT_SET; // Paper - Friction API
|
|
+
|
|
+ @Override
|
|
+ public float getBukkitYaw() {
|
|
+ return this.getYHeadRot();
|
|
+ }
|
|
+ // CraftBukkit end
|
|
|
|
protected LivingEntity(EntityType<? extends LivingEntity> entityType, Level level) {
|
|
super(entityType, level);
|
|
this.attributes = new AttributeMap(DefaultAttributes.getSupplier(entityType));
|
|
- this.setHealth(this.getMaxHealth());
|
|
+ this.craftAttributes = new CraftAttributeMap(this.attributes); // CraftBukkit
|
|
+ // CraftBukkit - this.setHealth(this.getMaxHealth()) inlined and simplified to skip the instanceof check for Player, as getBukkitEntity() is not initialized in constructor
|
|
+ this.entityData.set(LivingEntity.DATA_HEALTH_ID, this.getMaxHealth());
|
|
this.blocksBuilding = true;
|
|
this.rotA = (float)((Math.random() + 1.0) * 0.01F);
|
|
this.reapplyPosition();
|
|
@@ -360,7 +_,13 @@
|
|
float f = Mth.ceil(this.fallDistance - attributeValue);
|
|
double min = Math.min((double)(0.2F + f / 15.0F), 2.5);
|
|
int i = (int)(150.0 * min);
|
|
- serverLevel.sendParticles(new BlockParticleOption(ParticleTypes.BLOCK, state), x, y1, z, i, 0.0, 0.0, 0.0, 0.15F);
|
|
+ // CraftBukkit start - visiblity api
|
|
+ if (this instanceof ServerPlayer) {
|
|
+ serverLevel.sendParticlesSource((ServerPlayer) this, new BlockParticleOption(ParticleTypes.BLOCK, state), false, false, x, y1, z, i, 0.0, 0.0, 0.0, 0.15F);
|
|
+ } else {
|
|
+ serverLevel.sendParticles(new BlockParticleOption(ParticleTypes.BLOCK, state), x, y1, z, i, 0.0, 0.0, 0.0, 0.15F);
|
|
+ }
|
|
+ // CraftBukkit end
|
|
}
|
|
}
|
|
|
|
@@ -566,7 +_,7 @@
|
|
this.deathTime++;
|
|
if (this.deathTime >= 20 && !this.level().isClientSide() && !this.isRemoved()) {
|
|
this.level().broadcastEntityEvent(this, (byte)60);
|
|
- this.remove(Entity.RemovalReason.KILLED);
|
|
+ this.remove(Entity.RemovalReason.KILLED, EntityRemoveEvent.Cause.DEATH); // CraftBukkit - add Bukkit remove cause
|
|
}
|
|
}
|
|
|
|
@@ -658,7 +_,7 @@
|
|
}
|
|
|
|
public boolean shouldDiscardFriction() {
|
|
- return this.discardFriction;
|
|
+ return !this.frictionState.toBooleanOrElse(!this.discardFriction); // Paper - Friction API
|
|
}
|
|
|
|
public void setDiscardFriction(boolean discardFriction) {
|
|
@@ -670,11 +_,16 @@
|
|
}
|
|
|
|
public void onEquipItem(EquipmentSlot slot, ItemStack oldItem, ItemStack newItem) {
|
|
+ // CraftBukkit start
|
|
+ this.onEquipItem(slot, oldItem, newItem, false);
|
|
+ }
|
|
+ public void onEquipItem(EquipmentSlot slot, ItemStack oldItem, ItemStack newItem, boolean silent) {
|
|
+ // CraftBukkit end
|
|
if (!this.level().isClientSide() && !this.isSpectator()) {
|
|
boolean flag = newItem.isEmpty() && oldItem.isEmpty();
|
|
if (!flag && !ItemStack.isSameItemSameComponents(oldItem, newItem) && !this.firstTick) {
|
|
Equippable equippable = newItem.get(DataComponents.EQUIPPABLE);
|
|
- if (!this.isSilent() && equippable != null && slot == equippable.slot()) {
|
|
+ if (!this.isSilent() && equippable != null && slot == equippable.slot() && !silent) { // CraftBukkit
|
|
this.level()
|
|
.playSeededSound(
|
|
null, this.getX(), this.getY(), this.getZ(), equippable.equipSound(), this.getSoundSource(), 1.0F, 1.0F, this.random.nextLong()
|
|
@@ -690,11 +_,18 @@
|
|
|
|
@Override
|
|
public void remove(Entity.RemovalReason reason) {
|
|
+ // CraftBukkit start - add Bukkit remove cause
|
|
+ this.remove(reason, null);
|
|
+ }
|
|
+
|
|
+ @Override
|
|
+ public void remove(Entity.RemovalReason reason, EntityRemoveEvent.Cause eventCause) {
|
|
+ // CraftBukkit end
|
|
if ((reason == Entity.RemovalReason.KILLED || reason == Entity.RemovalReason.DISCARDED) && this.level() instanceof ServerLevel serverLevel) {
|
|
this.triggerOnDeathMobEffects(serverLevel, reason);
|
|
}
|
|
|
|
- super.remove(reason);
|
|
+ super.remove(reason, eventCause); // CraftBukkit
|
|
this.brain.clearMemories();
|
|
}
|
|
|
|
@@ -703,11 +_,17 @@
|
|
mobEffectInstance.onMobRemoved(level, this, removalReason);
|
|
}
|
|
|
|
+ this.removeAllEffects(org.bukkit.event.entity.EntityPotionEffectEvent.Cause.DEATH); // CraftBukkit
|
|
this.activeEffects.clear();
|
|
}
|
|
|
|
@Override
|
|
public void addAdditionalSaveData(CompoundTag compound) {
|
|
+ // Paper start - Friction API
|
|
+ if (this.frictionState != net.kyori.adventure.util.TriState.NOT_SET) {
|
|
+ compound.putString("Paper.FrictionState", this.frictionState.toString());
|
|
+ }
|
|
+ // Paper end - Friction API
|
|
compound.putFloat("Health", this.getHealth());
|
|
compound.putShort("HurtTime", (short)this.hurtTime);
|
|
compound.putInt("HurtByTimestamp", this.lastHurtByMobTimestamp);
|
|
@@ -736,7 +_,23 @@
|
|
|
|
@Override
|
|
public void readAdditionalSaveData(CompoundTag compound) {
|
|
- this.internalSetAbsorptionAmount(compound.getFloat("AbsorptionAmount"));
|
|
+ // Paper start - Check for NaN
|
|
+ float absorptionAmount = compound.getFloat("AbsorptionAmount");
|
|
+ if (Float.isNaN(absorptionAmount)) {
|
|
+ absorptionAmount = 0;
|
|
+ }
|
|
+ this.internalSetAbsorptionAmount(absorptionAmount);
|
|
+ // Paper end - Check for NaN
|
|
+ // Paper start - Friction API
|
|
+ if (compound.contains("Paper.FrictionState")) {
|
|
+ String frictionState = compound.getString("Paper.FrictionState");
|
|
+ try {
|
|
+ this.frictionState = net.kyori.adventure.util.TriState.valueOf(frictionState);
|
|
+ } catch (Exception ignored) {
|
|
+ LOGGER.error("Unknown friction state " + frictionState + " for " + this);
|
|
+ }
|
|
+ }
|
|
+ // Paper end - Friction API
|
|
if (compound.contains("attributes", 9) && this.level() != null && !this.level().isClientSide) {
|
|
this.getAttributes().load(compound.getList("attributes", 10));
|
|
}
|
|
@@ -753,6 +_,16 @@
|
|
}
|
|
}
|
|
|
|
+ // CraftBukkit start
|
|
+ if (compound.contains("Bukkit.MaxHealth")) {
|
|
+ Tag maxHealthTag = compound.get("Bukkit.MaxHealth");
|
|
+ if (maxHealthTag.getId() == 5) {
|
|
+ this.getAttribute(Attributes.MAX_HEALTH).setBaseValue(((FloatTag) maxHealthTag).getAsDouble());
|
|
+ } else if (maxHealthTag.getId() == 3) {
|
|
+ this.getAttribute(Attributes.MAX_HEALTH).setBaseValue(((IntTag) maxHealthTag).getAsDouble());
|
|
+ }
|
|
+ }
|
|
+ // CraftBukkit end
|
|
if (compound.contains("Health", 99)) {
|
|
this.setHealth(compound.getFloat("Health"));
|
|
}
|
|
@@ -764,6 +_,7 @@
|
|
String string = compound.getString("Team");
|
|
Scoreboard scoreboard = this.level().getScoreboard();
|
|
PlayerTeam playerTeam = scoreboard.getPlayerTeam(string);
|
|
+ if (!this.level().paperConfig().scoreboards.allowNonPlayerEntitiesOnScoreboards && !(this instanceof net.minecraft.world.entity.player.Player)) { playerTeam = null; } // Paper - Perf: Disable Scoreboards for non players by default
|
|
boolean flag = playerTeam != null && scoreboard.addPlayerToTeam(this.getStringUUID(), playerTeam);
|
|
if (!flag) {
|
|
LOGGER.warn("Unable to add mob to team \"{}\" (that team probably doesn't exist)", string);
|
|
@@ -776,11 +_,13 @@
|
|
|
|
if (compound.contains("SleepingX", 99) && compound.contains("SleepingY", 99) && compound.contains("SleepingZ", 99)) {
|
|
BlockPos blockPos = new BlockPos(compound.getInt("SleepingX"), compound.getInt("SleepingY"), compound.getInt("SleepingZ"));
|
|
+ if (this.position().distanceToSqr(blockPos.getX(), blockPos.getY(), blockPos.getZ()) < 16 * 16) { // Paper - The sleeping pos will always also set the actual pos, so a desync suggests something is wrong
|
|
this.setSleepingPos(blockPos);
|
|
this.entityData.set(DATA_POSE, Pose.SLEEPING);
|
|
if (!this.firstTick) {
|
|
this.setPosToBed(blockPos);
|
|
}
|
|
+ } // Paper - The sleeping pos will always also set the actual pos, so a desync suggests something is wrong
|
|
}
|
|
|
|
if (compound.contains("Brain", 10)) {
|
|
@@ -788,15 +_,44 @@
|
|
}
|
|
}
|
|
|
|
+ // CraftBukkit start
|
|
+ private boolean isTickingEffects = false;
|
|
+ private List<ProcessableEffect> effectsToProcess = Lists.newArrayList();
|
|
+
|
|
+ private static class ProcessableEffect {
|
|
+
|
|
+ private Holder<MobEffect> type;
|
|
+ private MobEffectInstance effect;
|
|
+ private final EntityPotionEffectEvent.Cause cause;
|
|
+
|
|
+ private ProcessableEffect(MobEffectInstance effect, EntityPotionEffectEvent.Cause cause) {
|
|
+ this.effect = effect;
|
|
+ this.cause = cause;
|
|
+ }
|
|
+
|
|
+ private ProcessableEffect(Holder<MobEffect> type, EntityPotionEffectEvent.Cause cause) {
|
|
+ this.type = type;
|
|
+ this.cause = cause;
|
|
+ }
|
|
+ }
|
|
+ // CraftBukkit end
|
|
+
|
|
protected void tickEffects() {
|
|
Iterator<Holder<MobEffect>> iterator = this.activeEffects.keySet().iterator();
|
|
|
|
+ this.isTickingEffects = true; // CraftBukkit
|
|
try {
|
|
while (iterator.hasNext()) {
|
|
Holder<MobEffect> holder = iterator.next();
|
|
MobEffectInstance mobEffectInstance = this.activeEffects.get(holder);
|
|
if (!mobEffectInstance.tick(this, () -> this.onEffectUpdated(mobEffectInstance, true, null))) {
|
|
if (!this.level().isClientSide) {
|
|
+ // CraftBukkit start
|
|
+ EntityPotionEffectEvent event = CraftEventFactory.callEntityPotionEffectChangeEvent(this, mobEffectInstance, null, org.bukkit.event.entity.EntityPotionEffectEvent.Cause.EXPIRATION);
|
|
+ if (event.isCancelled()) {
|
|
+ continue;
|
|
+ }
|
|
+ // CraftBukkit end
|
|
iterator.remove();
|
|
this.onEffectsRemoved(List.of(mobEffectInstance));
|
|
}
|
|
@@ -807,6 +_,17 @@
|
|
} catch (ConcurrentModificationException var6) {
|
|
}
|
|
|
|
+ // CraftBukkit start
|
|
+ this.isTickingEffects = false;
|
|
+ for (ProcessableEffect effect : this.effectsToProcess) {
|
|
+ if (effect.effect != null) {
|
|
+ this.addEffect(effect.effect, effect.cause);
|
|
+ } else {
|
|
+ this.removeEffect(effect.type, effect.cause);
|
|
+ }
|
|
+ }
|
|
+ this.effectsToProcess.clear();
|
|
+ // CraftBukkit end
|
|
if (this.effectsDirty) {
|
|
if (!this.level().isClientSide) {
|
|
this.updateInvisibilityStatus();
|
|
@@ -912,15 +_,33 @@
|
|
}
|
|
|
|
public boolean removeAllEffects() {
|
|
+ // CraftBukkit start
|
|
+ return this.removeAllEffects(org.bukkit.event.entity.EntityPotionEffectEvent.Cause.UNKNOWN);
|
|
+ }
|
|
+ public boolean removeAllEffects(EntityPotionEffectEvent.Cause cause) {
|
|
+ // CraftBukkit end
|
|
if (this.level().isClientSide) {
|
|
return false;
|
|
} else if (this.activeEffects.isEmpty()) {
|
|
return false;
|
|
} else {
|
|
- Map<Holder<MobEffect>, MobEffectInstance> map = Maps.newHashMap(this.activeEffects);
|
|
- this.activeEffects.clear();
|
|
- this.onEffectsRemoved(map.values());
|
|
- return true;
|
|
+ // CraftBukkit start
|
|
+ List<MobEffectInstance> toRemove = new LinkedList<>();
|
|
+ Iterator<MobEffectInstance> iterator = this.activeEffects.values().iterator();
|
|
+ while (iterator.hasNext()) {
|
|
+ MobEffectInstance effect = iterator.next();
|
|
+ EntityPotionEffectEvent event = CraftEventFactory.callEntityPotionEffectChangeEvent(this, effect, null, cause, EntityPotionEffectEvent.Action.CLEARED);
|
|
+ if (event.isCancelled()) {
|
|
+ continue;
|
|
+ }
|
|
+
|
|
+ iterator.remove();
|
|
+ toRemove.add(effect);
|
|
+ }
|
|
+
|
|
+ this.onEffectsRemoved(toRemove);
|
|
+ return !toRemove.isEmpty();
|
|
+ // CraftBukkit end
|
|
}
|
|
}
|
|
|
|
@@ -942,21 +_,57 @@
|
|
}
|
|
|
|
public final boolean addEffect(MobEffectInstance effectInstance) {
|
|
- return this.addEffect(effectInstance, null);
|
|
+ return this.addEffect(effectInstance, (Entity) null); // CraftBukkit
|
|
+ }
|
|
+
|
|
+ // CraftBukkit start
|
|
+ public boolean addEffect(MobEffectInstance effectInstance, EntityPotionEffectEvent.Cause cause) {
|
|
+ return this.addEffect(effectInstance, (Entity) null, cause);
|
|
}
|
|
|
|
public boolean addEffect(MobEffectInstance effectInstance, @Nullable Entity entity) {
|
|
+ return this.addEffect(effectInstance, entity, org.bukkit.event.entity.EntityPotionEffectEvent.Cause.UNKNOWN);
|
|
+ }
|
|
+
|
|
+ public boolean addEffect(MobEffectInstance effectInstance, @Nullable Entity entity, EntityPotionEffectEvent.Cause cause) {
|
|
+ // Paper start - Don't fire sync event during generation
|
|
+ return this.addEffect(effectInstance, entity, cause, true);
|
|
+ }
|
|
+ public boolean addEffect(MobEffectInstance effectInstance, @Nullable Entity entity, EntityPotionEffectEvent.Cause cause, boolean fireEvent) {
|
|
+ // Paper end - Don't fire sync event during generation
|
|
+ // org.spigotmc.AsyncCatcher.catchOp("effect add"); // Spigot // Paper - move to API
|
|
+ if (this.isTickingEffects) {
|
|
+ this.effectsToProcess.add(new ProcessableEffect(effectInstance, cause));
|
|
+ return true;
|
|
+ }
|
|
+ // CraftBukkit end
|
|
if (!this.canBeAffected(effectInstance)) {
|
|
return false;
|
|
} else {
|
|
MobEffectInstance mobEffectInstance = this.activeEffects.get(effectInstance.getEffect());
|
|
boolean flag = false;
|
|
+ // CraftBukkit start
|
|
+ boolean override = false;
|
|
+ if (mobEffectInstance != null) {
|
|
+ override = new MobEffectInstance(mobEffectInstance).update(effectInstance);
|
|
+ }
|
|
+
|
|
+ if (fireEvent) { // Paper - Don't fire sync event during generation
|
|
+ EntityPotionEffectEvent event = CraftEventFactory.callEntityPotionEffectChangeEvent(this, mobEffectInstance, effectInstance, cause, override);
|
|
+ override = event.isOverride(); // Paper - Don't fire sync event during generation
|
|
+ if (event.isCancelled()) {
|
|
+ return false;
|
|
+ }
|
|
+ } // Paper - Don't fire sync event during generation
|
|
+ // CraftBukkit end
|
|
if (mobEffectInstance == null) {
|
|
this.activeEffects.put(effectInstance.getEffect(), effectInstance);
|
|
this.onEffectAdded(effectInstance, entity);
|
|
flag = true;
|
|
effectInstance.onEffectAdded(this);
|
|
- } else if (mobEffectInstance.update(effectInstance)) {
|
|
+ // CraftBukkit start
|
|
+ } else if (override) { // Paper - Don't fire sync event during generation
|
|
+ mobEffectInstance.update(effectInstance);
|
|
this.onEffectUpdated(mobEffectInstance, true, entity);
|
|
flag = true;
|
|
}
|
|
@@ -995,11 +_,37 @@
|
|
|
|
@Nullable
|
|
public MobEffectInstance removeEffectNoUpdate(Holder<MobEffect> effect) {
|
|
+ // CraftBukkit start
|
|
+ return this.removeEffectNoUpdate(effect, org.bukkit.event.entity.EntityPotionEffectEvent.Cause.UNKNOWN);
|
|
+ }
|
|
+
|
|
+ @Nullable
|
|
+ public MobEffectInstance removeEffectNoUpdate(Holder<MobEffect> effect, EntityPotionEffectEvent.Cause cause) {
|
|
+ if (this.isTickingEffects) {
|
|
+ this.effectsToProcess.add(new ProcessableEffect(effect, cause));
|
|
+ return null;
|
|
+ }
|
|
+
|
|
+ MobEffectInstance effectInstance = this.activeEffects.get(effect);
|
|
+ if (effectInstance == null) {
|
|
+ return null;
|
|
+ }
|
|
+
|
|
+ EntityPotionEffectEvent event = CraftEventFactory.callEntityPotionEffectChangeEvent(this, effectInstance, null, cause);
|
|
+ if (event.isCancelled()) {
|
|
+ return null;
|
|
+ }
|
|
+
|
|
return this.activeEffects.remove(effect);
|
|
}
|
|
|
|
public boolean removeEffect(Holder<MobEffect> effect) {
|
|
- MobEffectInstance mobEffectInstance = this.removeEffectNoUpdate(effect);
|
|
+ return this.removeEffect(effect, org.bukkit.event.entity.EntityPotionEffectEvent.Cause.UNKNOWN);
|
|
+ }
|
|
+
|
|
+ public boolean removeEffect(Holder<MobEffect> effect, EntityPotionEffectEvent.Cause cause) {
|
|
+ MobEffectInstance mobEffectInstance = this.removeEffectNoUpdate(effect, cause);
|
|
+ // CraftBukkit end
|
|
if (mobEffectInstance != null) {
|
|
this.onEffectsRemoved(List.of(mobEffectInstance));
|
|
return true;
|
|
@@ -1080,17 +_,62 @@
|
|
}
|
|
|
|
public void heal(float healAmount) {
|
|
+ // CraftBukkit start - Delegate so we can handle providing a reason for health being regained
|
|
+ this.heal(healAmount, EntityRegainHealthEvent.RegainReason.CUSTOM);
|
|
+ }
|
|
+
|
|
+ public void heal(float healAmount, EntityRegainHealthEvent.RegainReason regainReason) {
|
|
+ // Paper start - Forward
|
|
+ this.heal(healAmount, regainReason, false);
|
|
+ }
|
|
+
|
|
+ public void heal(float healAmount, EntityRegainHealthEvent.RegainReason regainReason, boolean isFastRegen) {
|
|
+ // Paper end - Forward
|
|
float health = this.getHealth();
|
|
if (health > 0.0F) {
|
|
- this.setHealth(health + healAmount);
|
|
+ EntityRegainHealthEvent event = new EntityRegainHealthEvent(this.getBukkitEntity(), healAmount, regainReason, isFastRegen); // Paper
|
|
+ // Suppress during worldgen
|
|
+ if (this.valid) {
|
|
+ this.level().getCraftServer().getPluginManager().callEvent(event);
|
|
+ }
|
|
+
|
|
+ if (!event.isCancelled()) {
|
|
+ this.setHealth((float) (this.getHealth() + event.getAmount()));
|
|
+ }
|
|
+ // CraftBukkit end
|
|
}
|
|
}
|
|
|
|
public float getHealth() {
|
|
+ // CraftBukkit start - Use unscaled health
|
|
+ if (this instanceof ServerPlayer serverPlayer) {
|
|
+ return (float) serverPlayer.getBukkitEntity().getHealth();
|
|
+ }
|
|
+ // CraftBukkit end
|
|
return this.entityData.get(DATA_HEALTH_ID);
|
|
}
|
|
|
|
public void setHealth(float health) {
|
|
+ // Paper start - Check for NaN
|
|
+ if (Float.isNaN(health)) { health = getMaxHealth(); if (this.valid) {
|
|
+ System.err.println("[NAN-HEALTH] " + getScoreboardName() + " had NaN health set");
|
|
+ } } // Paper end - Check for NaN
|
|
+ // CraftBukkit start - Handle scaled health
|
|
+ if (this instanceof ServerPlayer) {
|
|
+ org.bukkit.craftbukkit.entity.CraftPlayer player = ((ServerPlayer) this).getBukkitEntity();
|
|
+ // Squeeze
|
|
+ if (health < 0.0F) {
|
|
+ player.setRealHealth(0.0D);
|
|
+ } else if (health > player.getMaxHealth()) {
|
|
+ player.setRealHealth(player.getMaxHealth());
|
|
+ } else {
|
|
+ player.setRealHealth(health);
|
|
+ }
|
|
+
|
|
+ player.updateScaledHealth(false);
|
|
+ return;
|
|
+ }
|
|
+ // CraftBukkit end
|
|
this.entityData.set(DATA_HEALTH_ID, Mth.clamp(health, 0.0F, this.getMaxHealth()));
|
|
}
|
|
|
|
@@ -1102,7 +_,7 @@
|
|
public boolean hurtServer(ServerLevel level, DamageSource damageSource, float amount) {
|
|
if (this.isInvulnerableTo(level, damageSource)) {
|
|
return false;
|
|
- } else if (this.isDeadOrDying()) {
|
|
+ } else if (this.isRemoved() || this.dead || this.getHealth() <= 0.0F) { // CraftBukkit - Don't allow entities that got set to dead/killed elsewhere to get damaged and die
|
|
return false;
|
|
} else if (damageSource.is(DamageTypeTags.IS_FIRE) && this.hasEffect(MobEffects.FIRE_RESISTANCE)) {
|
|
return false;
|
|
@@ -1116,47 +_,71 @@
|
|
amount = 0.0F;
|
|
}
|
|
|
|
- float f = amount;
|
|
- boolean flag = false;
|
|
+ float f = amount; final float originalAmount = amount; // Paper - revert to vanilla #hurt - OBFHELPER
|
|
+ boolean flag = amount > 0.0F && this.isDamageSourceBlocked(damageSource); // Copied from below;
|
|
float f1 = 0.0F;
|
|
- if (amount > 0.0F && this.isDamageSourceBlocked(damageSource)) {
|
|
+ // CraftBukkit - Moved into handleEntityDamage(DamageSource, float) for get f and actuallyHurt(DamageSource, float, EntityDamageEvent) for handle damage
|
|
+ if (false && amount > 0.0F && this.isDamageSourceBlocked(damageSource)) {
|
|
this.hurtCurrentlyUsedShield(amount);
|
|
f1 = amount;
|
|
amount = 0.0F;
|
|
- if (!damageSource.is(DamageTypeTags.IS_PROJECTILE) && damageSource.getDirectEntity() instanceof LivingEntity livingEntity) {
|
|
+ if (!damageSource.is(DamageTypeTags.IS_PROJECTILE) && damageSource.getDirectEntity() instanceof LivingEntity livingEntity && livingEntity.distanceToSqr(this) <= (200.0D * 200.0D)) { // Paper - Check distance in entity interactions
|
|
this.blockUsingShield(livingEntity);
|
|
}
|
|
|
|
flag = true;
|
|
}
|
|
|
|
- if (damageSource.is(DamageTypeTags.IS_FREEZING) && this.getType().is(EntityTypeTags.FREEZE_HURTS_EXTRA_TYPES)) {
|
|
+ // CraftBukkit - Moved into handleEntityDamage(DamageSource, float) for get f
|
|
+ if (false && damageSource.is(DamageTypeTags.IS_FREEZING) && this.getType().is(EntityTypeTags.FREEZE_HURTS_EXTRA_TYPES)) {
|
|
amount *= 5.0F;
|
|
}
|
|
|
|
- if (damageSource.is(DamageTypeTags.DAMAGES_HELMET) && !this.getItemBySlot(EquipmentSlot.HEAD).isEmpty()) {
|
|
+ // CraftBukkit - Moved into handleEntityDamage(DamageSource, float) for get f and actuallyHurt(DamageSource, float, EntityDamageEvent) for handle damage
|
|
+ if (false && damageSource.is(DamageTypeTags.DAMAGES_HELMET) && !this.getItemBySlot(EquipmentSlot.HEAD).isEmpty()) {
|
|
this.hurtHelmet(damageSource, amount);
|
|
amount *= 0.75F;
|
|
}
|
|
|
|
+ EntityDamageEvent event; // CraftBukkit // Paper - move this into the actual invuln check....
|
|
this.walkAnimation.setSpeed(1.5F);
|
|
if (Float.isNaN(amount) || Float.isInfinite(amount)) {
|
|
amount = Float.MAX_VALUE;
|
|
}
|
|
|
|
boolean flag1 = true;
|
|
- if (this.invulnerableTime > 10.0F && !damageSource.is(DamageTypeTags.BYPASSES_COOLDOWN)) {
|
|
+ if (this.invulnerableTime > (float) this.invulnerableDuration / 2.0F && !damageSource.is(DamageTypeTags.BYPASSES_COOLDOWN)) { // CraftBukkit - restore use of maxNoDamageTicks
|
|
if (amount <= this.lastHurt) {
|
|
return false;
|
|
}
|
|
|
|
- this.actuallyHurt(level, damageSource, amount - this.lastHurt);
|
|
+ // Paper start - only call damage event when actuallyHurt will be called - move call logic down
|
|
+ event = this.handleEntityDamage(damageSource, amount, this.lastHurt); // Paper - fix invulnerability reduction in EntityDamageEvent - pass lastDamage reduction
|
|
+ amount = computeAmountFromEntityDamageEvent(event);
|
|
+ // Paper end - only call damage event when actuallyHurt will be called - move call logic down
|
|
+
|
|
+ // CraftBukkit start
|
|
+ if (!this.actuallyHurt(level, damageSource, (float) event.getFinalDamage(), event)) { // Paper - fix invulnerability reduction in EntityDamageEvent - no longer subtract lastHurt, that is part of the damage event calc now
|
|
+ return false;
|
|
+ }
|
|
+ if (this instanceof ServerPlayer && event.getDamage() == 0 && originalAmount == 0) return false; // Paper - revert to vanilla damage - players are not affected by damage that is 0 - skip damage if the vanilla damage is 0 and was not modified by plugins in the event.
|
|
+ // CraftBukkit end
|
|
this.lastHurt = amount;
|
|
flag1 = false;
|
|
} else {
|
|
+ // Paper start - only call damage event when actuallyHurt will be called - move call logic down
|
|
+ event = this.handleEntityDamage(damageSource, amount, 0); // Paper - fix invulnerability reduction in EntityDamageEvent - pass lastDamage reduction (none in this branch)
|
|
+ amount = computeAmountFromEntityDamageEvent(event);
|
|
+ // Paper end - only call damage event when actuallyHurt will be called - move call logic down
|
|
+ // CraftBukkit start
|
|
+ if (!this.actuallyHurt(level, damageSource, (float) event.getFinalDamage(), event)) {
|
|
+ return false;
|
|
+ }
|
|
+ if (this instanceof ServerPlayer && event.getDamage() == 0 && originalAmount == 0) return false; // Paper - revert to vanilla damage - players are not affected by damage that is 0 - skip damage if the vanilla damage is 0 and was not modified by plugins in the event.
|
|
this.lastHurt = amount;
|
|
- this.invulnerableTime = 20;
|
|
- this.actuallyHurt(level, damageSource, amount);
|
|
+ this.invulnerableTime = this.invulnerableDuration; // CraftBukkit - restore use of maxNoDamageTicks
|
|
+ // this.actuallyHurt(level, damageSource, amount);
|
|
+ // CraftBukkit end
|
|
this.hurtDuration = 10;
|
|
this.hurtTime = this.hurtDuration;
|
|
}
|
|
@@ -1170,7 +_,7 @@
|
|
level.broadcastDamageEvent(this, damageSource);
|
|
}
|
|
|
|
- if (!damageSource.is(DamageTypeTags.NO_IMPACT) && (!flag || amount > 0.0F)) {
|
|
+ if (!damageSource.is(DamageTypeTags.NO_IMPACT) && !flag) { // CraftBukkit - Prevent marking hurt if the damage is blocked
|
|
this.markHurt();
|
|
}
|
|
|
|
@@ -1185,8 +_,16 @@
|
|
d = damageSource.getSourcePosition().x() - this.getX();
|
|
d1 = damageSource.getSourcePosition().z() - this.getZ();
|
|
}
|
|
+ // Paper start - Check distance in entity interactions; see for loop in knockback method
|
|
+ if (Math.abs(d) > 200) {
|
|
+ d = Math.random() - Math.random();
|
|
+ }
|
|
+ if (Math.abs(d1) > 200) {
|
|
+ d1 = Math.random() - Math.random();
|
|
+ }
|
|
+ // Paper end - Check distance in entity interactions
|
|
|
|
- this.knockback(0.4F, d, d1);
|
|
+ this.knockback(0.4F, d, d1, damageSource.getDirectEntity(), damageSource.getDirectEntity() == null ? io.papermc.paper.event.entity.EntityKnockbackEvent.Cause.DAMAGE : io.papermc.paper.event.entity.EntityKnockbackEvent.Cause.ENTITY_ATTACK); // CraftBukkit // Paper - knockback events
|
|
if (!flag) {
|
|
this.indicateDamage(d, d1);
|
|
}
|
|
@@ -1195,17 +_,18 @@
|
|
|
|
if (this.isDeadOrDying()) {
|
|
if (!this.checkTotemDeathProtection(damageSource)) {
|
|
- if (flag1) {
|
|
- this.makeSound(this.getDeathSound());
|
|
- }
|
|
+ // Paper start - moved into CraftEventFactory event caller for cancellable death event
|
|
+ this.silentDeath = !flag1; // mark entity as dying silently
|
|
+ // Paper end
|
|
|
|
this.die(damageSource);
|
|
+ this.silentDeath = false; // Paper - cancellable death event - reset to default
|
|
}
|
|
} else if (flag1) {
|
|
this.playHurtSound(damageSource);
|
|
}
|
|
|
|
- boolean flag2 = !flag || amount > 0.0F;
|
|
+ boolean flag2 = !flag; // CraftBukkit - Ensure to return false if damage is blocked
|
|
if (flag2) {
|
|
this.lastDamageSource = damageSource;
|
|
this.lastDamageStamp = this.level().getGameTime();
|
|
@@ -1259,12 +_,24 @@
|
|
}
|
|
}
|
|
|
|
+ // Paper start - only call damage event when actuallyHurt will be called - move out amount computation logic
|
|
+ private float computeAmountFromEntityDamageEvent(final EntityDamageEvent event) {
|
|
+ // Taken from hurt()'s craftbukkit diff.
|
|
+ float amount = 0;
|
|
+ amount += (float) event.getDamage(DamageModifier.BASE);
|
|
+ amount += (float) event.getDamage(DamageModifier.BLOCKING);
|
|
+ amount += (float) event.getDamage(DamageModifier.FREEZING);
|
|
+ amount += (float) event.getDamage(DamageModifier.HARD_HAT);
|
|
+ return amount;
|
|
+ }
|
|
+ // Paper end - only call damage event when actuallyHurt will be called - move out amount computation logic
|
|
+
|
|
protected void blockUsingShield(LivingEntity attacker) {
|
|
attacker.blockedByShield(this);
|
|
}
|
|
|
|
protected void blockedByShield(LivingEntity defender) {
|
|
- defender.knockback(0.5, defender.getX() - this.getX(), defender.getZ() - this.getZ());
|
|
+ defender.knockback(0.5, defender.getX() - this.getX(), defender.getZ() - this.getZ(), this, io.papermc.paper.event.entity.EntityKnockbackEvent.Cause.SHIELD_BLOCK); // CraftBukkit // Paper - fix attacker & knockback events
|
|
}
|
|
|
|
private boolean checkTotemDeathProtection(DamageSource damageSource) {
|
|
@@ -1274,18 +_,37 @@
|
|
ItemStack itemStack = null;
|
|
DeathProtection deathProtection = null;
|
|
|
|
+ // CraftBukkit start
|
|
+ InteractionHand hand = null;
|
|
+ ItemStack itemInHand = ItemStack.EMPTY;
|
|
for (InteractionHand interactionHand : InteractionHand.values()) {
|
|
- ItemStack itemInHand = this.getItemInHand(interactionHand);
|
|
+ itemInHand = this.getItemInHand(interactionHand);
|
|
deathProtection = itemInHand.get(DataComponents.DEATH_PROTECTION);
|
|
if (deathProtection != null) {
|
|
+ hand = interactionHand; // CraftBukkit
|
|
itemStack = itemInHand.copy();
|
|
+ // itemInHand.shrink(1); // CraftBukkit
|
|
+ break;
|
|
+ }
|
|
+ }
|
|
+
|
|
+ org.bukkit.inventory.EquipmentSlot handSlot = (hand != null) ? org.bukkit.craftbukkit.CraftEquipmentSlot.getHand(hand) : null;
|
|
+ EntityResurrectEvent event = new EntityResurrectEvent((org.bukkit.entity.LivingEntity) this.getBukkitEntity(), handSlot);
|
|
+ event.setCancelled(itemStack == null);
|
|
+ this.level().getCraftServer().getPluginManager().callEvent(event);
|
|
+
|
|
+ if (!event.isCancelled()) {
|
|
+ if (!itemInHand.isEmpty() && itemStack != null) { // Paper - only reduce item if actual totem was found
|
|
itemInHand.shrink(1);
|
|
- break;
|
|
- }
|
|
- }
|
|
-
|
|
- if (itemStack != null) {
|
|
- if (this instanceof ServerPlayer serverPlayer) {
|
|
+ }
|
|
+ // Paper start - fix NPE when pre-cancelled EntityResurrectEvent is uncancelled
|
|
+ // restore the previous behavior in that case by defaulting to vanillas totem of undying efect
|
|
+ if (deathProtection == null) {
|
|
+ deathProtection = DeathProtection.TOTEM_OF_UNDYING;
|
|
+ }
|
|
+ // Paper end - fix NPE when pre-cancelled EntityResurrectEvent is uncancelled
|
|
+ if (itemStack != null && this instanceof ServerPlayer serverPlayer) {
|
|
+ // CraftBukkit end
|
|
serverPlayer.awardStat(Stats.ITEM_USED.get(itemStack.getItem()));
|
|
CriteriaTriggers.USED_TOTEM.trigger(serverPlayer, itemStack);
|
|
this.gameEvent(GameEvent.ITEM_INTERACT_FINISH);
|
|
@@ -1364,6 +_,7 @@
|
|
if (!this.isRemoved() && !this.dead) {
|
|
Entity entity = damageSource.getEntity();
|
|
LivingEntity killCredit = this.getKillCredit();
|
|
+ /* // Paper - move down to make death event cancellable - this is the awardKillScore below
|
|
if (killCredit != null) {
|
|
killCredit.awardKillScore(this, damageSource);
|
|
}
|
|
@@ -1373,68 +_,142 @@
|
|
}
|
|
|
|
if (!this.level().isClientSide && this.hasCustomName()) {
|
|
- LOGGER.info("Named entity {} died: {}", this, this.getCombatTracker().getDeathMessage().getString());
|
|
+ if (org.spigotmc.SpigotConfig.logNamedDeaths) LivingEntity.LOGGER.info("Named entity {} died: {}", this, this.getCombatTracker().getDeathMessage().getString()); // Spigot
|
|
}
|
|
+ */ // Paper - move down to make death event cancellable - this is the awardKillScore below
|
|
|
|
this.dead = true;
|
|
- this.getCombatTracker().recheckStatus();
|
|
+ // Paper - moved into if below
|
|
if (this.level() instanceof ServerLevel serverLevel) {
|
|
- if (entity == null || entity.killedEntity(serverLevel, this)) {
|
|
+ // Paper - move below into if for onKill
|
|
+ // Paper start
|
|
+ org.bukkit.event.entity.EntityDeathEvent deathEvent = this.dropAllDeathLoot(serverLevel, damageSource);
|
|
+ if (deathEvent == null || !deathEvent.isCancelled()) {
|
|
+ //if (entityliving != null) { // Paper - Fix item duplication and teleport issues; moved to be run earlier in #dropAllDeathLoot before destroying the drop items in CraftEventFactory#callEntityDeathEvent
|
|
+ // entityliving.awardKillScore(this, damageSource);
|
|
+ //}
|
|
+ // Paper start - clear equipment if event is not cancelled
|
|
+ if (this instanceof Mob) {
|
|
+ for (EquipmentSlot slot : this.clearedEquipmentSlots) {
|
|
+ this.setItemSlot(slot, ItemStack.EMPTY);
|
|
+ }
|
|
+ this.clearedEquipmentSlots.clear();
|
|
+ }
|
|
+ // Paper end
|
|
+
|
|
+ if (this.isSleeping()) {
|
|
+ this.stopSleeping();
|
|
+ }
|
|
+
|
|
+ if (!this.level().isClientSide && this.hasCustomName()) {
|
|
+ if (org.spigotmc.SpigotConfig.logNamedDeaths) LivingEntity.LOGGER.info("Named entity {} died: {}", this, this.getCombatTracker().getDeathMessage().getString()); // Spigot
|
|
+ }
|
|
+
|
|
+ this.getCombatTracker().recheckStatus();
|
|
+ if (entity != null) {
|
|
+ entity.killedEntity((ServerLevel) this.level(), this);
|
|
+ }
|
|
this.gameEvent(GameEvent.ENTITY_DIE);
|
|
- this.dropAllDeathLoot(serverLevel, damageSource);
|
|
+ } else {
|
|
+ this.dead = false;
|
|
+ this.setHealth((float) deathEvent.getReviveHealth());
|
|
+ }
|
|
+ // Paper end
|
|
this.createWitherRose(killCredit);
|
|
}
|
|
|
|
+ // Paper start
|
|
+ if (this.dead) { // Paper
|
|
this.level().broadcastEntityEvent(this, (byte)3);
|
|
- }
|
|
|
|
this.setPose(Pose.DYING);
|
|
+ }
|
|
+ // Paper end
|
|
}
|
|
}
|
|
|
|
protected void createWitherRose(@Nullable LivingEntity entitySource) {
|
|
if (this.level() instanceof ServerLevel serverLevel) {
|
|
boolean var6 = false;
|
|
- if (entitySource instanceof WitherBoss) {
|
|
+ if (this.dead && entitySource instanceof WitherBoss) { // Paper
|
|
if (serverLevel.getGameRules().getBoolean(GameRules.RULE_MOBGRIEFING)) {
|
|
BlockPos blockPos = this.blockPosition();
|
|
BlockState blockState = Blocks.WITHER_ROSE.defaultBlockState();
|
|
if (this.level().getBlockState(blockPos).isAir() && blockState.canSurvive(this.level(), blockPos)) {
|
|
- this.level().setBlock(blockPos, blockState, 3);
|
|
- var6 = true;
|
|
+ var6 = org.bukkit.craftbukkit.event.CraftEventFactory.handleBlockFormEvent(this.level(), blockPos, blockState, 3, this); // CraftBukkit - call EntityBlockFormEvent for Wither Rose
|
|
}
|
|
}
|
|
|
|
if (!var6) {
|
|
ItemEntity itemEntity = new ItemEntity(this.level(), this.getX(), this.getY(), this.getZ(), new ItemStack(Items.WITHER_ROSE));
|
|
+ // CraftBukkit start
|
|
+ org.bukkit.event.entity.EntityDropItemEvent event = new org.bukkit.event.entity.EntityDropItemEvent(this.getBukkitEntity(), (org.bukkit.entity.Item) itemEntity.getBukkitEntity());
|
|
+ CraftEventFactory.callEvent(event);
|
|
+ if (event.isCancelled()) {
|
|
+ return;
|
|
+ }
|
|
+ // CraftBukkit end
|
|
this.level().addFreshEntity(itemEntity);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
- protected void dropAllDeathLoot(ServerLevel level, DamageSource damageSource) {
|
|
+ // Paper start
|
|
+ protected boolean clearEquipmentSlots = true;
|
|
+ protected Set<EquipmentSlot> clearedEquipmentSlots = new java.util.HashSet<>();
|
|
+ protected org.bukkit.event.entity.EntityDeathEvent dropAllDeathLoot(ServerLevel level, DamageSource damageSource) {
|
|
+ // Paper end
|
|
boolean flag = this.lastHurtByPlayerTime > 0;
|
|
+ this.dropEquipment(level); // CraftBukkit - from below
|
|
if (this.shouldDropLoot() && level.getGameRules().getBoolean(GameRules.RULE_DOMOBLOOT)) {
|
|
this.dropFromLootTable(level, damageSource, flag);
|
|
+ // Paper start
|
|
+ final boolean prev = this.clearEquipmentSlots;
|
|
+ this.clearEquipmentSlots = false;
|
|
+ this.clearedEquipmentSlots.clear();
|
|
+ // Paper end
|
|
this.dropCustomDeathLoot(level, damageSource, flag);
|
|
+ this.clearEquipmentSlots = prev; // Paper
|
|
}
|
|
|
|
- this.dropEquipment(level);
|
|
+ // CraftBukkit start - Call death event // Paper start - call advancement triggers with correct entity equipment
|
|
+ org.bukkit.event.entity.EntityDeathEvent deathEvent = CraftEventFactory.callEntityDeathEvent(this, damageSource, this.drops, () -> {
|
|
+ final LivingEntity entityliving = this.getKillCredit();
|
|
+ if (entityliving != null) {
|
|
+ entityliving.awardKillScore(this, damageSource);
|
|
+ }
|
|
+ }); // Paper end
|
|
+ this.postDeathDropItems(deathEvent); // Paper
|
|
+ this.drops = new ArrayList<>();
|
|
+ // this.dropEquipment(level); // CraftBukkit - moved up
|
|
+ // CraftBukkit end
|
|
this.dropExperience(level, damageSource.getEntity());
|
|
+ return deathEvent; // Paper
|
|
}
|
|
|
|
protected void dropEquipment(ServerLevel level) {
|
|
}
|
|
+ protected void postDeathDropItems(org.bukkit.event.entity.EntityDeathEvent event) {} // Paper - method for post death logic that cannot be ran before the event is potentially cancelled
|
|
|
|
- protected void dropExperience(ServerLevel level, @Nullable Entity entity) {
|
|
+ public int getExpReward(ServerLevel level, @Nullable Entity entity) { // CraftBukkit
|
|
if (!this.wasExperienceConsumed()
|
|
&& (
|
|
this.isAlwaysExperienceDropper()
|
|
|| this.lastHurtByPlayerTime > 0 && this.shouldDropExperience() && level.getGameRules().getBoolean(GameRules.RULE_DOMOBLOOT)
|
|
)) {
|
|
- ExperienceOrb.award(level, this.position(), this.getExperienceReward(level, entity));
|
|
- }
|
|
+ return this.getExperienceReward(level, entity); // CraftBukkit
|
|
+ }
|
|
+ return 0; // CraftBukkit
|
|
+ }
|
|
+
|
|
+ protected void dropExperience(ServerLevel level, @Nullable Entity entity) {
|
|
+ // CraftBukkit start - Update getExpReward() above if the removed if() changes!
|
|
+ if (!(this instanceof net.minecraft.world.entity.boss.enderdragon.EnderDragon)) { // CraftBukkit - SPIGOT-2420: Special case ender dragon will drop the xp over time
|
|
+ ExperienceOrb.award(level, this.position(), this.expToDrop, this instanceof ServerPlayer ? org.bukkit.entity.ExperienceOrb.SpawnReason.PLAYER_DEATH : org.bukkit.entity.ExperienceOrb.SpawnReason.ENTITY_DEATH, entity, this); // Paper
|
|
+ this.expToDrop = 0;
|
|
+ }
|
|
+ // CraftBukkit end
|
|
}
|
|
|
|
protected void dropCustomDeathLoot(ServerLevel level, DamageSource damageSource, boolean recentlyHit) {
|
|
@@ -1513,9 +_,14 @@
|
|
}
|
|
|
|
public void knockback(double strength, double x, double z) {
|
|
+ // CraftBukkit start - EntityKnockbackEvent
|
|
+ this.knockback(strength, x, z, null, io.papermc.paper.event.entity.EntityKnockbackEvent.Cause.UNKNOWN); // Paper - knockback events
|
|
+ }
|
|
+
|
|
+ public void knockback(double strength, double x, double z, @Nullable Entity attacker, io.papermc.paper.event.entity.EntityKnockbackEvent.Cause eventCause) { // Paper - knockback events
|
|
strength *= 1.0 - this.getAttributeValue(Attributes.KNOCKBACK_RESISTANCE);
|
|
- if (!(strength <= 0.0)) {
|
|
- this.hasImpulse = true;
|
|
+ if (true || !(strength <= 0.0)) { // CraftBukkit - Call event even when force is 0
|
|
+ // this.hasImpulse = true; // CraftBukkit - Move down
|
|
Vec3 deltaMovement = this.getDeltaMovement();
|
|
|
|
while (x * x + z * z < 1.0E-5F) {
|
|
@@ -1524,11 +_,22 @@
|
|
}
|
|
|
|
Vec3 vec3 = new Vec3(x, 0.0, z).normalize().scale(strength);
|
|
- this.setDeltaMovement(
|
|
+ // Paper start - knockback events
|
|
+ Vec3 finalVelocity = new Vec3(
|
|
deltaMovement.x / 2.0 - vec3.x,
|
|
this.onGround() ? Math.min(0.4, deltaMovement.y / 2.0 + strength) : deltaMovement.y,
|
|
deltaMovement.z / 2.0 - vec3.z
|
|
);
|
|
+ Vec3 diff = finalVelocity.subtract(deltaMovement);
|
|
+ io.papermc.paper.event.entity.EntityKnockbackEvent event = CraftEventFactory.callEntityKnockbackEvent((org.bukkit.craftbukkit.entity.CraftLivingEntity) this.getBukkitEntity(), attacker, attacker, eventCause, strength, diff);
|
|
+ // Paper end - knockback events
|
|
+ if (event.isCancelled()) {
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ this.hasImpulse = true;
|
|
+ this.setDeltaMovement(deltaMovement.add(event.getKnockback().getX(), event.getKnockback().getY(), event.getKnockback().getZ())); // Paper - knockback events
|
|
+ // CraftBukkit end
|
|
}
|
|
}
|
|
|
|
@@ -1584,6 +_,20 @@
|
|
return new LivingEntity.Fallsounds(SoundEvents.GENERIC_SMALL_FALL, SoundEvents.GENERIC_BIG_FALL);
|
|
}
|
|
|
|
+ // CraftBukkit start - Add delegate methods
|
|
+ public SoundEvent getHurtSound0(DamageSource damagesource) {
|
|
+ return this.getHurtSound(damagesource);
|
|
+ }
|
|
+
|
|
+ public SoundEvent getDeathSound0() {
|
|
+ return this.getDeathSound();
|
|
+ }
|
|
+
|
|
+ public SoundEvent getFallDamageSound0(int fallHeight) {
|
|
+ return this.getFallDamageSound(fallHeight);
|
|
+ }
|
|
+ // CraftBukkit end
|
|
+
|
|
public Optional<BlockPos> getLastClimbablePos() {
|
|
return this.lastClimbablePos;
|
|
}
|
|
@@ -1617,7 +_,7 @@
|
|
|
|
@Override
|
|
public boolean isAlive() {
|
|
- return !this.isRemoved() && this.getHealth() > 0.0F;
|
|
+ return !this.isRemoved() && this.getHealth() > 0.0F && !this.dead; // Paper - Check this.dead
|
|
}
|
|
|
|
public boolean isLookingAtMe(LivingEntity entity, double tolerance, boolean scaleByDistance, boolean visual, double... yValues) {
|
|
@@ -1651,9 +_,14 @@
|
|
boolean flag = super.causeFallDamage(fallDistance, multiplier, source);
|
|
int i = this.calculateFallDamage(fallDistance, multiplier);
|
|
if (i > 0) {
|
|
+ // CraftBukkit start
|
|
+ if (!this.hurtServer((ServerLevel) this.level(), source, (float) i)) {
|
|
+ return true;
|
|
+ }
|
|
+ // CraftBukkit end
|
|
this.playSound(this.getFallDamageSound(i), 1.0F, 1.0F);
|
|
this.playBlockFallSound();
|
|
- this.hurt(source, i);
|
|
+ // this.hurt(source, i); // CraftBukkit - moved up
|
|
return true;
|
|
} else {
|
|
return flag;
|
|
@@ -1718,7 +_,7 @@
|
|
|
|
protected float getDamageAfterArmorAbsorb(DamageSource damageSource, float damageAmount) {
|
|
if (!damageSource.is(DamageTypeTags.BYPASSES_ARMOR)) {
|
|
- this.hurtArmor(damageSource, damageAmount);
|
|
+ // this.hurtArmor(damageSource, damageAmount); // CraftBukkit - actuallyHurt(DamageSource, float, EntityDamageEvent) for damage handling
|
|
damageAmount = CombatRules.getDamageAfterAbsorb(
|
|
this, damageAmount, damageSource, this.getArmorValue(), (float)this.getAttributeValue(Attributes.ARMOR_TOUGHNESS)
|
|
);
|
|
@@ -1731,7 +_,8 @@
|
|
if (damageSource.is(DamageTypeTags.BYPASSES_EFFECTS)) {
|
|
return damageAmount;
|
|
} else {
|
|
- if (this.hasEffect(MobEffects.DAMAGE_RESISTANCE) && !damageSource.is(DamageTypeTags.BYPASSES_RESISTANCE)) {
|
|
+ // CraftBukkit - Moved to handleEntityDamage(DamageSource, float)
|
|
+ if (false && this.hasEffect(MobEffects.DAMAGE_RESISTANCE) && !damageSource.is(DamageTypeTags.BYPASSES_RESISTANCE)) {
|
|
int i = (this.getEffect(MobEffects.DAMAGE_RESISTANCE).getAmplifier() + 1) * 5;
|
|
int i1 = 25 - i;
|
|
float f = damageAmount * i1;
|
|
@@ -1768,24 +_,212 @@
|
|
}
|
|
}
|
|
|
|
- protected void actuallyHurt(ServerLevel level, DamageSource damageSource, float amount) {
|
|
+ // CraftBukkit start
|
|
+ private EntityDamageEvent handleEntityDamage(final DamageSource damagesource, float amount, final float invulnerabilityRelatedLastDamage) { // Paper - fix invulnerability reduction in EntityDamageEvent
|
|
+ float originalDamage = amount;
|
|
+ // Paper start - fix invulnerability reduction in EntityDamageEvent
|
|
+ final com.google.common.base.Function<Double, Double> invulnerabilityReductionEquation = d -> {
|
|
+ if (invulnerabilityRelatedLastDamage == 0) return 0D; // no last damage, no reduction
|
|
+ // last damage existed, this means the reduction *technically* is (new damage - last damage).
|
|
+ // If the event damage was changed to something less than invul damage, hard lock it at 0.
|
|
+ if (d < invulnerabilityRelatedLastDamage) return 0D;
|
|
+ return (double) -invulnerabilityRelatedLastDamage;
|
|
+ };
|
|
+ final float originalInvulnerabilityReduction = invulnerabilityReductionEquation.apply((double) amount).floatValue();
|
|
+ amount += originalInvulnerabilityReduction;
|
|
+ // Paper end - fix invulnerability reduction in EntityDamageEvent
|
|
+
|
|
+ com.google.common.base.Function<Double, Double> freezing = new com.google.common.base.Function<Double, Double>() {
|
|
+ @Override
|
|
+ public Double apply(Double f) {
|
|
+ if (damagesource.is(DamageTypeTags.IS_FREEZING) && LivingEntity.this.getType().is(EntityTypeTags.FREEZE_HURTS_EXTRA_TYPES)) {
|
|
+ return -(f - (f * 5.0F));
|
|
+ }
|
|
+ return -0.0;
|
|
+ }
|
|
+ };
|
|
+ float freezingModifier = freezing.apply((double) amount).floatValue();
|
|
+ amount += freezingModifier;
|
|
+
|
|
+ com.google.common.base.Function<Double, Double> hardHat = new com.google.common.base.Function<Double, Double>() {
|
|
+ @Override
|
|
+ public Double apply(Double f) {
|
|
+ if (damagesource.is(DamageTypeTags.DAMAGES_HELMET) && !LivingEntity.this.getItemBySlot(EquipmentSlot.HEAD).isEmpty()) {
|
|
+ return -(f - (f * 0.75F));
|
|
+ }
|
|
+ return -0.0;
|
|
+ }
|
|
+ };
|
|
+ float hardHatModifier = hardHat.apply((double) amount).floatValue();
|
|
+ amount += hardHatModifier;
|
|
+
|
|
+ com.google.common.base.Function<Double, Double> blocking = new com.google.common.base.Function<Double, Double>() {
|
|
+ @Override
|
|
+ public Double apply(Double f) {
|
|
+ return -((LivingEntity.this.isDamageSourceBlocked(damagesource)) ? f : 0.0);
|
|
+ }
|
|
+ };
|
|
+ float blockingModifier = blocking.apply((double) amount).floatValue();
|
|
+ amount += blockingModifier;
|
|
+
|
|
+ com.google.common.base.Function<Double, Double> armor = new com.google.common.base.Function<Double, Double>() {
|
|
+ @Override
|
|
+ public Double apply(Double f) {
|
|
+ return -(f - LivingEntity.this.getDamageAfterArmorAbsorb(damagesource, f.floatValue()));
|
|
+ }
|
|
+ };
|
|
+ float armorModifier = armor.apply((double) amount).floatValue();
|
|
+ amount += armorModifier;
|
|
+
|
|
+ com.google.common.base.Function<Double, Double> resistance = new com.google.common.base.Function<Double, Double>() {
|
|
+ @Override
|
|
+ public Double apply(Double f) {
|
|
+ if (!damagesource.is(DamageTypeTags.BYPASSES_EFFECTS) && LivingEntity.this.hasEffect(MobEffects.DAMAGE_RESISTANCE) && !damagesource.is(DamageTypeTags.BYPASSES_RESISTANCE)) {
|
|
+ int i = (LivingEntity.this.getEffect(MobEffects.DAMAGE_RESISTANCE).getAmplifier() + 1) * 5;
|
|
+ int j = 25 - i;
|
|
+ float f1 = f.floatValue() * (float) j;
|
|
+
|
|
+ return -(f - Math.max(f1 / 25.0F, 0.0F));
|
|
+ }
|
|
+ return -0.0;
|
|
+ }
|
|
+ };
|
|
+ float resistanceModifier = resistance.apply((double) amount).floatValue();
|
|
+ amount += resistanceModifier;
|
|
+
|
|
+ com.google.common.base.Function<Double, Double> magic = new com.google.common.base.Function<Double, Double>() {
|
|
+ @Override
|
|
+ public Double apply(Double f) {
|
|
+ return -(f - LivingEntity.this.getDamageAfterMagicAbsorb(damagesource, f.floatValue()));
|
|
+ }
|
|
+ };
|
|
+ float magicModifier = magic.apply((double) amount).floatValue();
|
|
+ amount += magicModifier;
|
|
+
|
|
+ com.google.common.base.Function<Double, Double> absorption = new com.google.common.base.Function<Double, Double>() {
|
|
+ @Override
|
|
+ public Double apply(Double f) {
|
|
+ return -(Math.max(f - Math.max(f - LivingEntity.this.getAbsorptionAmount(), 0.0F), 0.0F));
|
|
+ }
|
|
+ };
|
|
+ float absorptionModifier = absorption.apply((double) amount).floatValue();
|
|
+
|
|
+ // Paper start - fix invulnerability reduction in EntityDamageEvent
|
|
+ return CraftEventFactory.handleLivingEntityDamageEvent(this, damagesource, originalDamage, freezingModifier, hardHatModifier, blockingModifier, armorModifier, resistanceModifier, magicModifier, absorptionModifier, freezing, hardHat, blocking, armor, resistance, magic, absorption, (damageModifierDoubleMap, damageModifierFunctionMap) -> {
|
|
+ damageModifierFunctionMap.put(DamageModifier.INVULNERABILITY_REDUCTION, invulnerabilityReductionEquation);
|
|
+ damageModifierDoubleMap.put(DamageModifier.INVULNERABILITY_REDUCTION, (double) originalInvulnerabilityReduction);
|
|
+ });
|
|
+ // Paper end - fix invulnerability reduction in EntityDamageEvent
|
|
+ }
|
|
+
|
|
+ protected boolean actuallyHurt(ServerLevel level, final DamageSource damageSource, float amount, final EntityDamageEvent event) { // void -> boolean, add final
|
|
if (!this.isInvulnerableTo(level, damageSource)) {
|
|
- amount = this.getDamageAfterArmorAbsorb(damageSource, amount);
|
|
- amount = this.getDamageAfterMagicAbsorb(damageSource, amount);
|
|
- float var10 = Math.max(amount - this.getAbsorptionAmount(), 0.0F);
|
|
- this.setAbsorptionAmount(this.getAbsorptionAmount() - (amount - var10));
|
|
- float f1 = amount - var10;
|
|
+ if (event.isCancelled()) {
|
|
+ return false;
|
|
+ }
|
|
+
|
|
+ if (damageSource.getEntity() instanceof net.minecraft.world.entity.player.Player) {
|
|
+ // Paper start - PlayerAttackEntityCooldownResetEvent
|
|
+ //((net.minecraft.world.entity.player.Player) damagesource.getEntity()).resetAttackStrengthTicker(); // Moved from EntityHuman in order to make the cooldown reset get called after the damage event is fired
|
|
+ if (damageSource.getEntity() instanceof ServerPlayer) {
|
|
+ ServerPlayer player = (ServerPlayer) damageSource.getEntity();
|
|
+ if (new com.destroystokyo.paper.event.player.PlayerAttackEntityCooldownResetEvent(player.getBukkitEntity(), this.getBukkitEntity(), player.getAttackStrengthScale(0F)).callEvent()) {
|
|
+ player.resetAttackStrengthTicker();
|
|
+ }
|
|
+ } else {
|
|
+ ((net.minecraft.world.entity.player.Player) damageSource.getEntity()).resetAttackStrengthTicker();
|
|
+ }
|
|
+ // Paper end - PlayerAttackEntityCooldownResetEvent
|
|
+ }
|
|
+
|
|
+ // Resistance
|
|
+ if (event.getDamage(DamageModifier.RESISTANCE) < 0) {
|
|
+ float f3 = (float) -event.getDamage(DamageModifier.RESISTANCE);
|
|
+ if (f3 > 0.0F && f3 < 3.4028235E37F) {
|
|
+ if (this instanceof ServerPlayer) {
|
|
+ ((ServerPlayer) this).awardStat(Stats.DAMAGE_RESISTED, Math.round(f3 * 10.0F));
|
|
+ } else if (damageSource.getEntity() instanceof ServerPlayer) {
|
|
+ ((ServerPlayer) damageSource.getEntity()).awardStat(Stats.DAMAGE_DEALT_RESISTED, Math.round(f3 * 10.0F));
|
|
+ }
|
|
+ }
|
|
+ }
|
|
+
|
|
+ // Apply damage to helmet
|
|
+ if (damageSource.is(DamageTypeTags.DAMAGES_HELMET) && !this.getItemBySlot(EquipmentSlot.HEAD).isEmpty()) {
|
|
+ this.hurtHelmet(damageSource, amount);
|
|
+ }
|
|
+
|
|
+ // Apply damage to armor
|
|
+ if (!damageSource.is(DamageTypeTags.BYPASSES_ARMOR)) {
|
|
+ float armorDamage = (float) (event.getDamage() + event.getDamage(DamageModifier.BLOCKING) + event.getDamage(DamageModifier.HARD_HAT));
|
|
+ this.hurtArmor(damageSource, armorDamage);
|
|
+ }
|
|
+
|
|
+ // Apply blocking code // PAIL: steal from above
|
|
+ if (event.getDamage(DamageModifier.BLOCKING) < 0) {
|
|
+ this.hurtCurrentlyUsedShield((float) -event.getDamage(DamageModifier.BLOCKING));
|
|
+ Entity entity = damageSource.getDirectEntity();
|
|
+
|
|
+ if (!damageSource.is(DamageTypeTags.IS_PROJECTILE) && entity instanceof LivingEntity && entity.distanceToSqr(this) <= (200.0D * 200.0D)) { // Paper - Fix shield disable inconsistency & Check distance in entity interactions
|
|
+ this.blockUsingShield((LivingEntity) entity);
|
|
+ }
|
|
+ }
|
|
+
|
|
+ boolean human = this instanceof net.minecraft.world.entity.player.Player;
|
|
+ float originalDamage = (float) event.getDamage();
|
|
+ float absorptionModifier = (float) -event.getDamage(DamageModifier.ABSORPTION);
|
|
+ this.setAbsorptionAmount(Math.max(this.getAbsorptionAmount() - absorptionModifier, 0.0F));
|
|
+ float f1 = absorptionModifier;
|
|
+
|
|
+ if (f1 > 0.0F && f1 < 3.4028235E37F && this instanceof Player player) {
|
|
+ player.awardStat(Stats.DAMAGE_ABSORBED, Math.round(f1 * 10.0F));
|
|
+ }
|
|
+ // CraftBukkit end
|
|
if (f1 > 0.0F && f1 < 3.4028235E37F && damageSource.getEntity() instanceof ServerPlayer serverPlayer) {
|
|
serverPlayer.awardStat(Stats.DAMAGE_DEALT_ABSORBED, Math.round(f1 * 10.0F));
|
|
}
|
|
|
|
- if (var10 != 0.0F) {
|
|
- this.getCombatTracker().recordDamage(damageSource, var10);
|
|
- this.setHealth(this.getHealth() - var10);
|
|
- this.setAbsorptionAmount(this.getAbsorptionAmount() - var10);
|
|
+ // CraftBukkit start
|
|
+ if (amount > 0 || !human) {
|
|
+ if (human) {
|
|
+ // PAIL: Be sure to drag all this code from the EntityHuman subclass each update.
|
|
+ ((net.minecraft.world.entity.player.Player) this).causeFoodExhaustion(damageSource.getFoodExhaustion(), org.bukkit.event.entity.EntityExhaustionEvent.ExhaustionReason.DAMAGED); // CraftBukkit - EntityExhaustionEvent
|
|
+ if (amount < 3.4028235E37F) {
|
|
+ ((net.minecraft.world.entity.player.Player) this).awardStat(Stats.DAMAGE_TAKEN, Math.round(amount * 10.0F));
|
|
+ }
|
|
+ }
|
|
+ // CraftBukkit end
|
|
+ this.getCombatTracker().recordDamage(damageSource, amount);
|
|
+ this.setHealth(this.getHealth() - amount);
|
|
+ // CraftBukkit start
|
|
+ if (!human) {
|
|
+ this.setAbsorptionAmount(this.getAbsorptionAmount() - amount);
|
|
+ }
|
|
this.gameEvent(GameEvent.ENTITY_DAMAGE);
|
|
+ return true;
|
|
+ } else {
|
|
+ // Duplicate triggers if blocking
|
|
+ if (event.getDamage(DamageModifier.BLOCKING) < 0) {
|
|
+ if (this instanceof ServerPlayer) {
|
|
+ CriteriaTriggers.ENTITY_HURT_PLAYER.trigger((ServerPlayer) this, damageSource, originalDamage, amount, true); // Paper - fix taken/dealt param order
|
|
+ f1 = (float) -event.getDamage(DamageModifier.BLOCKING);
|
|
+ if (f1 > 0.0F && f1 < 3.4028235E37F) {
|
|
+ ((ServerPlayer) this).awardStat(Stats.DAMAGE_BLOCKED_BY_SHIELD, Math.round(originalDamage * 10.0F));
|
|
+ }
|
|
+ }
|
|
+
|
|
+ if (damageSource.getEntity() instanceof ServerPlayer) {
|
|
+ CriteriaTriggers.PLAYER_HURT_ENTITY.trigger((ServerPlayer) damageSource.getEntity(), this, damageSource, originalDamage, amount, true); // Paper - fix taken/dealt param order
|
|
+ }
|
|
+
|
|
+ return !io.papermc.paper.configuration.GlobalConfiguration.get().unsupportedSettings.skipVanillaDamageTickWhenShieldBlocked; // Paper - this should always return true, however expose an unsupported setting to flip this to false to enable "shield stunning".
|
|
+ } else {
|
|
+ return true; // Paper - return false ONLY if event was cancelled
|
|
+ }
|
|
+ // CraftBukkit end
|
|
}
|
|
}
|
|
+ return true; // CraftBukkit // Paper - return false ONLY if event was cancelled
|
|
}
|
|
|
|
public CombatTracker getCombatTracker() {
|
|
@@ -1814,7 +_,17 @@
|
|
}
|
|
|
|
public final void setArrowCount(int count) {
|
|
- this.entityData.set(DATA_ARROW_COUNT_ID, count);
|
|
+ // CraftBukkit start
|
|
+ this.setArrowCount(count, false);
|
|
+ }
|
|
+
|
|
+ public final void setArrowCount(int count, boolean reset) {
|
|
+ ArrowBodyCountChangeEvent event = CraftEventFactory.callArrowBodyCountChangeEvent(this, this.getArrowCount(), count, reset);
|
|
+ if (event.isCancelled()) {
|
|
+ return;
|
|
+ }
|
|
+ this.entityData.set(DATA_ARROW_COUNT_ID, event.getNewAmount());
|
|
+ // CraftBukkit end
|
|
}
|
|
|
|
public final int getStingerCount() {
|
|
@@ -1957,7 +_,7 @@
|
|
|
|
@Override
|
|
protected void onBelowWorld() {
|
|
- this.hurt(this.damageSources().fellOutOfWorld(), 4.0F);
|
|
+ this.hurt(this.damageSources().fellOutOfWorld(), this.level().getWorld().getVoidDamageAmount()); // Paper - use configured void damage amount
|
|
}
|
|
|
|
protected void updateSwingTime() {
|
|
@@ -2052,6 +_,12 @@
|
|
|
|
public abstract ItemStack getItemBySlot(EquipmentSlot slot);
|
|
|
|
+ // CraftBukkit start
|
|
+ public void setItemSlot(EquipmentSlot enumitemslot, ItemStack itemstack, boolean silent) {
|
|
+ this.setItemSlot(enumitemslot, itemstack);
|
|
+ }
|
|
+ // CraftBukkit end
|
|
+
|
|
public abstract void setItemSlot(EquipmentSlot slot, ItemStack stack);
|
|
|
|
public Iterable<ItemStack> getHandSlots() {
|
|
@@ -2158,14 +_,27 @@
|
|
return this.hasEffect(MobEffects.JUMP) ? 0.1F * (this.getEffect(MobEffects.JUMP).getAmplifier() + 1.0F) : 0.0F;
|
|
}
|
|
|
|
+ protected long lastJumpTime = 0L; // Paper - Prevent excessive velocity through repeated crits
|
|
@VisibleForTesting
|
|
public void jumpFromGround() {
|
|
float jumpPower = this.getJumpPower();
|
|
if (!(jumpPower <= 1.0E-5F)) {
|
|
Vec3 deltaMovement = this.getDeltaMovement();
|
|
+ // Paper start - Prevent excessive velocity through repeated crits
|
|
+ long time = System.nanoTime();
|
|
+ boolean canCrit = true;
|
|
+ if (this instanceof net.minecraft.world.entity.player.Player) {
|
|
+ canCrit = false;
|
|
+ if (time - this.lastJumpTime > (long)(0.250e9)) {
|
|
+ this.lastJumpTime = time;
|
|
+ canCrit = true;
|
|
+ }
|
|
+ }
|
|
+ // Paper end - Prevent excessive velocity through repeated crits
|
|
this.setDeltaMovement(deltaMovement.x, Math.max((double)jumpPower, deltaMovement.y), deltaMovement.z);
|
|
if (this.isSprinting()) {
|
|
float f = this.getYRot() * (float) (Math.PI / 180.0);
|
|
+ if (canCrit) // Paper - Prevent excessive velocity through repeated crits
|
|
this.addDeltaMovement(new Vec3(-Mth.sin(f) * 0.2, 0.0, Mth.cos(f) * 0.2));
|
|
}
|
|
|
|
@@ -2425,7 +_,7 @@
|
|
}
|
|
|
|
protected float getFlyingSpeed() {
|
|
- return this.getControllingPassenger() instanceof Player ? this.getSpeed() * 0.1F : 0.02F;
|
|
+ return this.getControllingPassenger() instanceof net.minecraft.world.entity.player.Player ? this.getSpeed() * 0.1F : 0.02F;
|
|
}
|
|
|
|
public float getSpeed() {
|
|
@@ -2471,7 +_,7 @@
|
|
}
|
|
}
|
|
|
|
- this.detectEquipmentUpdates();
|
|
+ this.detectEquipmentUpdatesPublic(); // CraftBukkit
|
|
if (this.tickCount % 20 == 0) {
|
|
this.getCombatTracker().recheckStatus();
|
|
}
|
|
@@ -2519,37 +_,14 @@
|
|
profilerFiller.pop();
|
|
profilerFiller.push("rangeChecks");
|
|
|
|
- while (this.getYRot() - this.yRotO < -180.0F) {
|
|
- this.yRotO -= 360.0F;
|
|
- }
|
|
-
|
|
- while (this.getYRot() - this.yRotO >= 180.0F) {
|
|
- this.yRotO += 360.0F;
|
|
- }
|
|
-
|
|
- while (this.yBodyRot - this.yBodyRotO < -180.0F) {
|
|
- this.yBodyRotO -= 360.0F;
|
|
- }
|
|
-
|
|
- while (this.yBodyRot - this.yBodyRotO >= 180.0F) {
|
|
- this.yBodyRotO += 360.0F;
|
|
- }
|
|
-
|
|
- while (this.getXRot() - this.xRotO < -180.0F) {
|
|
- this.xRotO -= 360.0F;
|
|
- }
|
|
-
|
|
- while (this.getXRot() - this.xRotO >= 180.0F) {
|
|
- this.xRotO += 360.0F;
|
|
- }
|
|
-
|
|
- while (this.yHeadRot - this.yHeadRotO < -180.0F) {
|
|
- this.yHeadRotO -= 360.0F;
|
|
- }
|
|
-
|
|
- while (this.yHeadRot - this.yHeadRotO >= 180.0F) {
|
|
- this.yHeadRotO += 360.0F;
|
|
- }
|
|
+ // Paper start - stop large pitch and yaw changes from crashing the server
|
|
+ this.yRotO += Math.round((this.getYRot() - this.yRotO) / 360.0F) * 360.0F;
|
|
+
|
|
+ this.yBodyRotO += Math.round((this.yBodyRot - this.yBodyRotO) / 360.0F) * 360.0F;
|
|
+
|
|
+ this.xRotO += Math.round((this.getXRot() - this.xRotO) / 360.0F) * 360.0F;
|
|
+
|
|
+ this.yHeadRotO += Math.round((this.yHeadRot - this.yHeadRotO) / 360.0F) * 360.0F;
|
|
|
|
profilerFiller.pop();
|
|
this.animStep += f2;
|
|
@@ -2573,7 +_,7 @@
|
|
this.elytraAnimationState.tick();
|
|
}
|
|
|
|
- public void detectEquipmentUpdates() {
|
|
+ public void detectEquipmentUpdatesPublic() { // CraftBukkit
|
|
Map<EquipmentSlot, ItemStack> map = this.collectEquipmentChanges();
|
|
if (map != null) {
|
|
this.handleHandSwap(map);
|
|
@@ -2595,6 +_,13 @@
|
|
};
|
|
ItemStack itemBySlot = this.getItemBySlot(equipmentSlot);
|
|
if (this.equipmentHasChanged(itemStack, itemBySlot)) {
|
|
+ // Paper start - PlayerArmorChangeEvent
|
|
+ if (this instanceof ServerPlayer && equipmentSlot.getType() == EquipmentSlot.Type.HUMANOID_ARMOR) {
|
|
+ final org.bukkit.inventory.ItemStack oldItem = CraftItemStack.asBukkitCopy(itemStack);
|
|
+ final org.bukkit.inventory.ItemStack newItem = CraftItemStack.asBukkitCopy(itemBySlot);
|
|
+ new com.destroystokyo.paper.event.player.PlayerArmorChangeEvent((org.bukkit.entity.Player) this.getBukkitEntity(), com.destroystokyo.paper.event.player.PlayerArmorChangeEvent.SlotType.valueOf(equipmentSlot.name()), oldItem, newItem).callEvent();
|
|
+ }
|
|
+ // Paper end - PlayerArmorChangeEvent
|
|
if (map == null) {
|
|
map = Maps.newEnumMap(EquipmentSlot.class);
|
|
}
|
|
@@ -2664,7 +_,7 @@
|
|
this.lastBodyItemStack = itemStack;
|
|
}
|
|
});
|
|
- ((ServerLevel)this.level()).getChunkSource().broadcast(this, new ClientboundSetEquipmentPacket(this.getId(), list));
|
|
+ ((ServerLevel)this.level()).getChunkSource().broadcast(this, new ClientboundSetEquipmentPacket(this.getId(), list, true)); // Paper - data sanitization
|
|
}
|
|
|
|
private ItemStack getLastArmorItem(EquipmentSlot slot) {
|
|
@@ -2765,8 +_,10 @@
|
|
if (!flag || this.onGround() && !(fluidHeight > fluidJumpThreshold)) {
|
|
if (!this.isInLava() || this.onGround() && !(fluidHeight > fluidJumpThreshold)) {
|
|
if ((this.onGround() || flag && fluidHeight <= fluidJumpThreshold) && this.noJumpDelay == 0) {
|
|
+ if (new com.destroystokyo.paper.event.entity.EntityJumpEvent(getBukkitLivingEntity()).callEvent()) { // Paper - Entity Jump API
|
|
this.jumpFromGround();
|
|
this.noJumpDelay = 10;
|
|
+ } else { this.setJumping(false); } // Paper - Entity Jump API; setJumping(false) stops a potential loop
|
|
}
|
|
} else {
|
|
this.jumpInLiquid(FluidTags.LAVA);
|
|
@@ -2805,7 +_,7 @@
|
|
this.calculateEntityAnimation(this instanceof FlyingAnimal);
|
|
profilerFiller.pop();
|
|
profilerFiller.push("freezing");
|
|
- if (!this.level().isClientSide && !this.isDeadOrDying()) {
|
|
+ if (!this.level().isClientSide && !this.isDeadOrDying() && !this.freezeLocked) { // Paper - Freeze Tick Lock API
|
|
int ticksFrozen = this.getTicksFrozen();
|
|
if (this.isInPowderSnow && this.canFreeze()) {
|
|
this.setTicksFrozen(Math.min(this.getTicksRequiredToFreeze(), ticksFrozen + 1));
|
|
@@ -2829,6 +_,20 @@
|
|
|
|
this.pushEntities();
|
|
profilerFiller.pop();
|
|
+ // Paper start - Add EntityMoveEvent
|
|
+ if (((ServerLevel) this.level()).hasEntityMoveEvent && !(this instanceof Player)) {
|
|
+ if (this.xo != this.getX() || this.yo != this.getY() || this.zo != this.getZ() || this.yRotO != this.getYRot() || this.xRotO != this.getXRot()) {
|
|
+ Location from = new Location(this.level().getWorld(), this.xo, this.yo, this.zo, this.yRotO, this.xRotO);
|
|
+ Location to = new Location(this.level().getWorld(), this.getX(), this.getY(), this.getZ(), this.getYRot(), this.getXRot());
|
|
+ io.papermc.paper.event.entity.EntityMoveEvent event = new io.papermc.paper.event.entity.EntityMoveEvent(this.getBukkitLivingEntity(), from, to.clone());
|
|
+ if (!event.callEvent()) {
|
|
+ this.absMoveTo(from.getX(), from.getY(), from.getZ(), from.getYaw(), from.getPitch());
|
|
+ } else if (!to.equals(event.getTo())) {
|
|
+ this.absMoveTo(event.getTo().getX(), event.getTo().getY(), event.getTo().getZ(), event.getTo().getYaw(), event.getTo().getPitch());
|
|
+ }
|
|
+ }
|
|
+ }
|
|
+ // Paper end - Add EntityMoveEvent
|
|
if (this.level() instanceof ServerLevel serverLevel && this.isSensitiveToWater() && this.isInWaterRainOrBubble()) {
|
|
this.hurtServer(serverLevel, this.damageSources().drown(), 1.0F);
|
|
}
|
|
@@ -2842,6 +_,7 @@
|
|
this.checkSlowFallDistance();
|
|
if (!this.level().isClientSide) {
|
|
if (!this.canGlide()) {
|
|
+ if (this.getSharedFlag(7) != false && !CraftEventFactory.callToggleGlideEvent(this, false).isCancelled()) // CraftBukkit
|
|
this.setSharedFlag(7, false);
|
|
return;
|
|
}
|
|
@@ -2881,9 +_,24 @@
|
|
if (!(this.level() instanceof ServerLevel serverLevel)) {
|
|
this.level().getEntities(EntityTypeTest.forClass(Player.class), this.getBoundingBox(), EntitySelector.pushableBy(this)).forEach(this::doPush);
|
|
} else {
|
|
- List<Entity> entities = this.level().getEntities(this, this.getBoundingBox(), EntitySelector.pushableBy(this));
|
|
+ // Paper start - don't run getEntities if we're not going to use its result
|
|
+ if (!this.isPushable()) {
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ net.minecraft.world.scores.Team team = this.getTeam();
|
|
+ if (team != null && team.getCollisionRule() == net.minecraft.world.scores.Team.CollisionRule.NEVER) {
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ int _int = serverLevel.getGameRules().getInt(GameRules.RULE_MAX_ENTITY_CRAMMING);
|
|
+ if (_int <= 0 && this.level().paperConfig().collisions.maxEntityCollisions <= 0) {
|
|
+ return;
|
|
+ }
|
|
+ // Paper end - don't run getEntities if we're not going to use its result
|
|
+ List<Entity> entities = this.level().getEntities(this, this.getBoundingBox(), EntitySelector.pushable(this, this.level().paperConfig().collisions.fixClimbingBypassingCrammingRule)); // Paper - Climbing should not bypass cramming gamerule
|
|
if (!entities.isEmpty()) {
|
|
- int _int = serverLevel.getGameRules().getInt(GameRules.RULE_MAX_ENTITY_CRAMMING);
|
|
+ // Paper - don't run getEntities if we're not going to use its result; moved up
|
|
if (_int > 0 && entities.size() > _int - 1 && this.random.nextInt(4) == 0) {
|
|
int i = 0;
|
|
|
|
@@ -2898,7 +_,16 @@
|
|
}
|
|
}
|
|
|
|
+ // Paper start - Cap entity collisions
|
|
+ this.numCollisions = Math.max(0, this.numCollisions - this.level().paperConfig().collisions.maxEntityCollisions);
|
|
for (Entity entity1 : entities) {
|
|
+ if (this.numCollisions >= this.level().paperConfig().collisions.maxEntityCollisions) {
|
|
+ break;
|
|
+ }
|
|
+
|
|
+ entity1.numCollisions++;
|
|
+ this.numCollisions++;
|
|
+ // Paper end - Cap entity collisions
|
|
this.doPush(entity1);
|
|
}
|
|
}
|
|
@@ -2941,9 +_,16 @@
|
|
|
|
@Override
|
|
public void stopRiding() {
|
|
+ // Paper start - Force entity dismount during teleportation
|
|
+ this.stopRiding(false);
|
|
+ }
|
|
+
|
|
+ @Override
|
|
+ public void stopRiding(boolean suppressCancellation) {
|
|
+ // Paper end - Force entity dismount during teleportation
|
|
Entity vehicle = this.getVehicle();
|
|
- super.stopRiding();
|
|
- if (vehicle != null && vehicle != this.getVehicle() && !this.level().isClientSide) {
|
|
+ super.stopRiding(suppressCancellation); // Paper - Force entity dismount during teleportation
|
|
+ if (vehicle != null && vehicle != this.getVehicle() && !this.level().isClientSide && vehicle.valid) { // Paper - don't process on world gen
|
|
this.dismountVehicle(vehicle);
|
|
}
|
|
}
|
|
@@ -3007,7 +_,7 @@
|
|
}
|
|
|
|
public void onItemPickup(ItemEntity itemEntity) {
|
|
- Entity owner = itemEntity.getOwner();
|
|
+ Entity owner = itemEntity.thrower != null ? this.level().getGlobalPlayerByUUID(itemEntity.thrower) : null; // Paper - check global player list where appropriate
|
|
if (owner instanceof ServerPlayer) {
|
|
CriteriaTriggers.THROWN_ITEM_PICKED_UP_BY_ENTITY.trigger((ServerPlayer)owner, itemEntity.getItem(), this);
|
|
}
|
|
@@ -3017,7 +_,7 @@
|
|
if (!entity.isRemoved()
|
|
&& !this.level().isClientSide
|
|
&& (entity instanceof ItemEntity || entity instanceof AbstractArrow || entity instanceof ExperienceOrb)) {
|
|
- ((ServerLevel)this.level()).getChunkSource().broadcast(entity, new ClientboundTakeItemEntityPacket(entity.getId(), this.getId(), amount));
|
|
+ ((ServerLevel)this.level()).getChunkSource().broadcastAndSend(this, new ClientboundTakeItemEntityPacket(entity.getId(), this.getId(), amount)); // Paper - broadcast with collector as source
|
|
}
|
|
}
|
|
|
|
@@ -3031,7 +_,8 @@
|
|
} else {
|
|
Vec3 vec3 = new Vec3(this.getX(), this.getEyeY(), this.getZ());
|
|
Vec3 vec31 = new Vec3(entity.getX(), y, entity.getZ());
|
|
- return !(vec31.distanceTo(vec3) > 128.0) && this.level().clip(new ClipContext(vec3, vec31, block, fluid, this)).getType() == HitResult.Type.MISS;
|
|
+ // Paper - diff on change - used in CraftLivingEntity#hasLineOfSight(Location) and CraftWorld#lineOfSightExists
|
|
+ return !(vec31.distanceToSqr(vec3) > 128.0D * 128.0D) && this.level().clip(new ClipContext(vec3, vec31, block, fluid, this)).getType() == HitResult.Type.MISS; // Paper - Perf: Use distance squared
|
|
}
|
|
}
|
|
|
|
@@ -3051,13 +_,27 @@
|
|
|
|
@Override
|
|
public boolean isPickable() {
|
|
- return !this.isRemoved();
|
|
+ return !this.isRemoved() && this.collides; // CraftBukkit
|
|
}
|
|
|
|
+ // Paper start - Climbing should not bypass cramming gamerule
|
|
@Override
|
|
public boolean isPushable() {
|
|
- return this.isAlive() && !this.isSpectator() && !this.onClimbable();
|
|
- }
|
|
+ return this.isCollidable(this.level().paperConfig().collisions.fixClimbingBypassingCrammingRule);
|
|
+ }
|
|
+
|
|
+ @Override
|
|
+ public boolean isCollidable(boolean ignoreClimbing) {
|
|
+ return this.isAlive() && !this.isSpectator() && (ignoreClimbing || !this.onClimbable()) && this.collides; // CraftBukkit
|
|
+ // Paper end - Climbing should not bypass cramming gamerule
|
|
+ }
|
|
+
|
|
+ // CraftBukkit start - collidable API
|
|
+ @Override
|
|
+ public boolean canCollideWithBukkit(Entity entity) {
|
|
+ return this.isPushable() && this.collides != this.collidableExemptions.contains(entity.getUUID());
|
|
+ }
|
|
+ // CraftBukkit end
|
|
|
|
@Override
|
|
public float getYHeadRot() {
|
|
@@ -3088,7 +_,7 @@
|
|
}
|
|
|
|
public final void setAbsorptionAmount(float absorptionAmount) {
|
|
- this.internalSetAbsorptionAmount(Mth.clamp(absorptionAmount, 0.0F, this.getMaxAbsorption()));
|
|
+ this.internalSetAbsorptionAmount(!Float.isNaN(absorptionAmount) ? Mth.clamp(absorptionAmount, 0.0F, this.getMaxAbsorption()) : 0.0F); // Paper - Check for NaN
|
|
}
|
|
|
|
protected void internalSetAbsorptionAmount(float absorptionAmount) {
|
|
@@ -3115,6 +_,15 @@
|
|
return (this.entityData.get(DATA_LIVING_ENTITY_FLAGS) & 2) > 0 ? InteractionHand.OFF_HAND : InteractionHand.MAIN_HAND;
|
|
}
|
|
|
|
+ // Paper start - Properly cancel usable items
|
|
+ public void resyncUsingItem(ServerPlayer serverPlayer) {
|
|
+ this.resendPossiblyDesyncedDataValues(java.util.List.of(DATA_LIVING_ENTITY_FLAGS), serverPlayer);
|
|
+ }
|
|
+ // Paper end - Properly cancel usable items
|
|
+ // Paper start - lag compensate eating
|
|
+ protected long eatStartTime;
|
|
+ protected int totalEatTimeTicks;
|
|
+ // Paper end - lag compensate eating
|
|
private void updatingUsingItem() {
|
|
if (this.isUsingItem()) {
|
|
if (ItemStack.isSameItem(this.getItemInHand(this.getUsedItemHand()), this.useItem)) {
|
|
@@ -3128,7 +_,12 @@
|
|
|
|
protected void updateUsingItem(ItemStack usingItem) {
|
|
usingItem.onUseTick(this.level(), this, this.getUseItemRemainingTicks());
|
|
- if (--this.useItemRemaining == 0 && !this.level().isClientSide && !usingItem.useOnRelease()) {
|
|
+ // Paper start - lag compensate eating
|
|
+ // we add 1 to the expected time to avoid lag compensating when we should not
|
|
+ final boolean shouldLagCompensate = this.useItem.has(DataComponents.FOOD) && this.eatStartTime != -1 && (System.nanoTime() - this.eatStartTime) > ((1L + this.totalEatTimeTicks) * 50L * (1000L * 1000L));
|
|
+ if ((--this.useItemRemaining == 0 || shouldLagCompensate) && !this.level().isClientSide && !usingItem.useOnRelease()) {
|
|
+ this.useItemRemaining = 0;
|
|
+ // Paper end - lag compensate eating
|
|
this.completeUsingItem();
|
|
}
|
|
}
|
|
@@ -3154,10 +_,18 @@
|
|
}
|
|
|
|
public void startUsingItem(InteractionHand hand) {
|
|
+ // Paper start - Prevent consuming the wrong itemstack
|
|
+ this.startUsingItem(hand, false);
|
|
+ }
|
|
+ public void startUsingItem(InteractionHand hand, boolean forceUpdate) {
|
|
+ // Paper end - Prevent consuming the wrong itemstack
|
|
ItemStack itemInHand = this.getItemInHand(hand);
|
|
- if (!itemInHand.isEmpty() && !this.isUsingItem()) {
|
|
+ if (!itemInHand.isEmpty() && !this.isUsingItem() || forceUpdate) { // Paper - Prevent consuming the wrong itemstack
|
|
this.useItem = itemInHand;
|
|
- this.useItemRemaining = itemInHand.getUseDuration(this);
|
|
+ // Paper start - lag compensate eating
|
|
+ this.useItemRemaining = this.totalEatTimeTicks = itemInHand.getUseDuration(this);
|
|
+ this.eatStartTime = System.nanoTime();
|
|
+ // Paper end - lag compensate eating
|
|
if (!this.level().isClientSide) {
|
|
this.setLivingEntityFlag(1, true);
|
|
this.setLivingEntityFlag(2, hand == InteractionHand.OFF_HAND);
|
|
@@ -3181,7 +_,10 @@
|
|
}
|
|
} else if (!this.isUsingItem() && !this.useItem.isEmpty()) {
|
|
this.useItem = ItemStack.EMPTY;
|
|
- this.useItemRemaining = 0;
|
|
+ // Paper start - lag compensate eating
|
|
+ this.useItemRemaining = this.totalEatTimeTicks = 0;
|
|
+ this.eatStartTime = -1L;
|
|
+ // Paper end - lag compensate eating
|
|
}
|
|
}
|
|
}
|
|
@@ -3220,12 +_,49 @@
|
|
this.releaseUsingItem();
|
|
} else {
|
|
if (!this.useItem.isEmpty() && this.isUsingItem()) {
|
|
- ItemStack itemStack = this.useItem.finishUsingItem(this.level(), this);
|
|
+ this.startUsingItem(this.getUsedItemHand(), true); // Paper - Prevent consuming the wrong itemstack
|
|
+ // CraftBukkit start - fire PlayerItemConsumeEvent
|
|
+ ItemStack itemStack;
|
|
+ PlayerItemConsumeEvent event = null; // Paper
|
|
+ if (this instanceof ServerPlayer serverPlayer) {
|
|
+ org.bukkit.inventory.ItemStack craftItem = CraftItemStack.asBukkitCopy(this.useItem);
|
|
+ org.bukkit.inventory.EquipmentSlot hand = org.bukkit.craftbukkit.CraftEquipmentSlot.getHand(usedItemHand);
|
|
+ event = new PlayerItemConsumeEvent((org.bukkit.entity.Player) this.getBukkitEntity(), craftItem, hand); // Paper
|
|
+ this.level().getCraftServer().getPluginManager().callEvent(event);
|
|
+
|
|
+ if (event.isCancelled()) {
|
|
+ // Update client
|
|
+ Consumable consumable = this.useItem.get(DataComponents.CONSUMABLE);
|
|
+ if (consumable != null) {
|
|
+ consumable.cancelUsingItem(serverPlayer, this.useItem);
|
|
+ }
|
|
+ serverPlayer.getBukkitEntity().updateInventory();
|
|
+ serverPlayer.getBukkitEntity().updateScaledHealth();
|
|
+ this.stopUsingItem(); // Paper - event is using an item, clear active item to reset its use
|
|
+ return;
|
|
+ }
|
|
+
|
|
+ itemStack = (craftItem.equals(event.getItem())) ? this.useItem.finishUsingItem(this.level(), this) : CraftItemStack.asNMSCopy(event.getItem()).finishUsingItem(this.level(), this);
|
|
+ } else {
|
|
+ itemStack = this.useItem.finishUsingItem(this.level(), this);
|
|
+ }
|
|
+ // Paper start - save the default replacement item and change it if necessary
|
|
+ final ItemStack defaultReplacement = itemStack;
|
|
+ if (event != null && event.getReplacement() != null) {
|
|
+ itemStack = CraftItemStack.asNMSCopy(event.getReplacement());
|
|
+ }
|
|
+ // Paper end
|
|
+ // CraftBukkit end
|
|
if (itemStack != this.useItem) {
|
|
this.setItemInHand(usedItemHand, itemStack);
|
|
}
|
|
|
|
this.stopUsingItem();
|
|
+ // Paper start
|
|
+ if (this instanceof ServerPlayer) {
|
|
+ ((ServerPlayer) this).getBukkitEntity().updateInventory();
|
|
+ }
|
|
+ // Paper end
|
|
}
|
|
}
|
|
}
|
|
@@ -3248,6 +_,7 @@
|
|
|
|
public void releaseUsingItem() {
|
|
if (!this.useItem.isEmpty()) {
|
|
+ if (this instanceof ServerPlayer) new io.papermc.paper.event.player.PlayerStopUsingItemEvent((org.bukkit.entity.Player) getBukkitEntity(), useItem.asBukkitMirror(), getTicksUsingItem()).callEvent(); // Paper - Add PlayerStopUsingItemEvent
|
|
this.useItem.releaseUsing(this.level(), this, this.getUseItemRemainingTicks());
|
|
if (this.useItem.useOnRelease()) {
|
|
this.updatingUsingItem();
|
|
@@ -3267,7 +_,10 @@
|
|
}
|
|
|
|
this.useItem = ItemStack.EMPTY;
|
|
- this.useItemRemaining = 0;
|
|
+ // Paper start - lag compensate eating
|
|
+ this.useItemRemaining = this.totalEatTimeTicks = 0;
|
|
+ this.eatStartTime = -1L;
|
|
+ // Paper end - lag compensate eating
|
|
}
|
|
|
|
public boolean isBlocking() {
|
|
@@ -3281,12 +_,69 @@
|
|
if (item.getUseAnimation(this.useItem) != ItemUseAnimation.BLOCK) {
|
|
return null;
|
|
} else {
|
|
- return item.getUseDuration(this.useItem, this) - this.useItemRemaining < 5 ? null : this.useItem;
|
|
+ return item.getUseDuration(this.useItem, this) - this.useItemRemaining < this.getShieldBlockingDelay() ? null : this.useItem; // Paper - Make shield blocking delay configurable
|
|
}
|
|
} else {
|
|
return null;
|
|
}
|
|
}
|
|
+
|
|
+ // Paper start - Make shield blocking delay configurable
|
|
+ public HitResult getRayTrace(int maxDistance, ClipContext.Fluid fluidCollisionOption) {
|
|
+ if (maxDistance < 1 || maxDistance > 120) {
|
|
+ throw new IllegalArgumentException("maxDistance must be between 1-120");
|
|
+ }
|
|
+
|
|
+ Vec3 start = new Vec3(getX(), getY() + getEyeHeight(), getZ());
|
|
+ org.bukkit.util.Vector dir = getBukkitEntity().getLocation().getDirection().multiply(maxDistance);
|
|
+ Vec3 end = new Vec3(start.x + dir.getX(), start.y + dir.getY(), start.z + dir.getZ());
|
|
+ ClipContext raytrace = new ClipContext(start, end, ClipContext.Block.OUTLINE, fluidCollisionOption, this);
|
|
+
|
|
+ return this.level().clip(raytrace);
|
|
+ }
|
|
+
|
|
+ public @Nullable net.minecraft.world.phys.EntityHitResult getTargetEntity(int maxDistance) {
|
|
+ if (maxDistance < 1 || maxDistance > 120) {
|
|
+ throw new IllegalArgumentException("maxDistance must be between 1-120");
|
|
+ }
|
|
+
|
|
+ Vec3 start = this.getEyePosition(1.0F);
|
|
+ Vec3 direction = this.getLookAngle();
|
|
+ Vec3 end = start.add(direction.x * maxDistance, direction.y * maxDistance, direction.z * maxDistance);
|
|
+
|
|
+ List<Entity> entityList = this.level().getEntities(this, getBoundingBox().expandTowards(direction.x * maxDistance, direction.y * maxDistance, direction.z * maxDistance).inflate(1.0D, 1.0D, 1.0D), EntitySelector.NO_SPECTATORS.and(Entity::isPickable));
|
|
+
|
|
+ double distance = 0.0D;
|
|
+ net.minecraft.world.phys.EntityHitResult result = null;
|
|
+
|
|
+ for (Entity entity : entityList) {
|
|
+ final double inflationAmount = (double) entity.getPickRadius();
|
|
+ AABB aabb = entity.getBoundingBox().inflate(inflationAmount, inflationAmount, inflationAmount);
|
|
+ Optional<Vec3> rayTraceResult = aabb.clip(start, end);
|
|
+
|
|
+ if (rayTraceResult.isPresent()) {
|
|
+ Vec3 rayTrace = rayTraceResult.get();
|
|
+ double distanceTo = start.distanceToSqr(rayTrace);
|
|
+ if (distanceTo < distance || distance == 0.0D) {
|
|
+ result = new net.minecraft.world.phys.EntityHitResult(entity, rayTrace);
|
|
+ distance = distanceTo;
|
|
+ }
|
|
+ }
|
|
+ }
|
|
+
|
|
+ return result;
|
|
+ }
|
|
+
|
|
+ public int shieldBlockingDelay = this.level().paperConfig().misc.shieldBlockingDelay;
|
|
+
|
|
+ public int getShieldBlockingDelay() {
|
|
+ return shieldBlockingDelay;
|
|
+ }
|
|
+
|
|
+ public void setShieldBlockingDelay(int shieldBlockingDelay) {
|
|
+ this.shieldBlockingDelay = shieldBlockingDelay;
|
|
+ }
|
|
+ // Paper end - Make shield blocking delay configurable
|
|
|
|
public boolean isSuppressingSlidingDownLadder() {
|
|
return this.isShiftKeyDown();
|
|
@@ -3306,6 +_,12 @@
|
|
}
|
|
|
|
public boolean randomTeleport(double x, double y, double z, boolean broadcastTeleport) {
|
|
+ // CraftBukkit start
|
|
+ return this.randomTeleport(x, y, z, broadcastTeleport, org.bukkit.event.player.PlayerTeleportEvent.TeleportCause.UNKNOWN).orElse(false);
|
|
+ }
|
|
+
|
|
+ public Optional<Boolean> randomTeleport(double x, double y, double z, boolean broadcastTeleport, org.bukkit.event.player.PlayerTeleportEvent.TeleportCause cause) {
|
|
+ // CraftBukkit end
|
|
double x1 = this.getX();
|
|
double y1 = this.getY();
|
|
double z1 = this.getZ();
|
|
@@ -3328,16 +_,39 @@
|
|
}
|
|
|
|
if (flag1) {
|
|
- this.teleportTo(x, d, z);
|
|
+ // CraftBukkit start - Teleport event
|
|
+ // first set position, to check if the place to teleport is valid
|
|
+ this.setPos(x, d, z);
|
|
if (level.noCollision(this) && !level.containsAnyLiquid(this.getBoundingBox())) {
|
|
flag = true;
|
|
}
|
|
+ // now revert and call event if the teleport place is valid
|
|
+ this.setPos(x1, y1, z1);
|
|
+
|
|
+ if (flag) {
|
|
+ if (!(this instanceof ServerPlayer)) {
|
|
+ EntityTeleportEvent teleport = new EntityTeleportEvent(this.getBukkitEntity(), new Location(this.level().getWorld(), x1, y1, z1), new Location(this.level().getWorld(), x, d, z));
|
|
+ this.level().getCraftServer().getPluginManager().callEvent(teleport);
|
|
+ if (!teleport.isCancelled() && teleport.getTo() != null) { // Paper
|
|
+ Location to = teleport.getTo();
|
|
+ this.teleportTo(to.getX(), to.getY(), to.getZ());
|
|
+ } else {
|
|
+ return Optional.empty();
|
|
+ }
|
|
+ } else {
|
|
+ // player teleport event is called in the underlining code
|
|
+ if (!((ServerPlayer) this).connection.teleport(x, d, z, this.getYRot(), this.getXRot(), cause)) {
|
|
+ return Optional.empty();
|
|
+ }
|
|
+ }
|
|
+ }
|
|
+ // CraftBukkit end
|
|
}
|
|
}
|
|
|
|
if (!flag) {
|
|
- this.teleportTo(x1, y1, z1);
|
|
- return false;
|
|
+ // this.teleportTo(x1, y1, z1); // CraftBukkit - already set the location back
|
|
+ return Optional.of(false); // CraftBukkit
|
|
} else {
|
|
if (broadcastTeleport) {
|
|
level.broadcastEntityEvent(this, (byte)46);
|
|
@@ -3347,7 +_,7 @@
|
|
pathfinderMob.getNavigation().stop();
|
|
}
|
|
|
|
- return true;
|
|
+ return Optional.of(true); // CraftBukkit
|
|
}
|
|
}
|
|
|