package net.minecraft.world.entity.ai.goal; import java.util.EnumSet; import net.minecraft.util.Mth; import net.minecraft.world.entity.LivingEntity; import net.minecraft.world.entity.Mob; import net.minecraft.world.entity.monster.RangedAttackMob; import org.jetbrains.annotations.Nullable; public class RangedAttackGoal extends Goal { private final Mob mob; private final RangedAttackMob rangedAttackMob; @Nullable private LivingEntity target; private int attackTime = -1; private final double speedModifier; private int seeTime; private final int attackIntervalMin; private final int attackIntervalMax; private final float attackRadius; private final float attackRadiusSqr; public RangedAttackGoal(RangedAttackMob rangedAttackMob, double speedModifier, int attackInterval, float attackRadius) { this(rangedAttackMob, speedModifier, attackInterval, attackInterval, attackRadius); } public RangedAttackGoal(RangedAttackMob rangedAttackMob, double speedModifier, int attackIntervalMin, int attackIntervalMax, float attackRadius) { if (!(rangedAttackMob instanceof LivingEntity)) { throw new IllegalArgumentException("ArrowAttackGoal requires Mob implements RangedAttackMob"); } else { this.rangedAttackMob = rangedAttackMob; this.mob = (Mob)rangedAttackMob; this.speedModifier = speedModifier; this.attackIntervalMin = attackIntervalMin; this.attackIntervalMax = attackIntervalMax; this.attackRadius = attackRadius; this.attackRadiusSqr = attackRadius * attackRadius; this.setFlags(EnumSet.of(Goal.Flag.MOVE, Goal.Flag.LOOK)); } } @Override public boolean canUse() { LivingEntity livingEntity = this.mob.getTarget(); if (livingEntity != null && livingEntity.isAlive()) { this.target = livingEntity; return true; } else { return false; } } @Override public boolean canContinueToUse() { return this.canUse() || this.target.isAlive() && !this.mob.getNavigation().isDone(); } @Override public void stop() { this.target = null; this.seeTime = 0; this.attackTime = -1; } @Override public boolean requiresUpdateEveryTick() { return true; } @Override public void tick() { double d = this.mob.distanceToSqr(this.target.getX(), this.target.getY(), this.target.getZ()); boolean bl = this.mob.getSensing().hasLineOfSight(this.target); if (bl) { this.seeTime++; } else { this.seeTime = 0; } if (!(d > this.attackRadiusSqr) && this.seeTime >= 5) { this.mob.getNavigation().stop(); } else { this.mob.getNavigation().moveTo(this.target, this.speedModifier); } this.mob.getLookControl().setLookAt(this.target, 30.0F, 30.0F); if (--this.attackTime == 0) { if (!bl) { return; } float f = (float)Math.sqrt(d) / this.attackRadius; float g = Mth.clamp(f, 0.1F, 1.0F); this.rangedAttackMob.performRangedAttack(this.target, g); this.attackTime = Mth.floor(f * (this.attackIntervalMax - this.attackIntervalMin) + this.attackIntervalMin); } else if (this.attackTime < 0) { this.attackTime = Mth.floor(Mth.lerp(Math.sqrt(d) / this.attackRadius, (double)this.attackIntervalMin, (double)this.attackIntervalMax)); } } }