100 lines
4.1 KiB
Java
100 lines
4.1 KiB
Java
package net.minecraft.world.entity.ai.goal;
|
|
|
|
import java.util.EnumSet;
|
|
import java.util.List;
|
|
import java.util.function.Predicate;
|
|
import javax.annotation.Nullable;
|
|
import net.minecraft.world.entity.Mob;
|
|
import net.minecraft.world.entity.ai.control.LookControl;
|
|
import net.minecraft.world.entity.ai.navigation.FlyingPathNavigation;
|
|
import net.minecraft.world.entity.ai.navigation.GroundPathNavigation;
|
|
import net.minecraft.world.entity.ai.navigation.PathNavigation;
|
|
import net.minecraft.world.level.pathfinder.PathType;
|
|
|
|
public class FollowMobGoal extends Goal {
|
|
private final Mob mob;
|
|
private final Predicate<Mob> followPredicate;
|
|
@Nullable
|
|
private Mob followingMob;
|
|
private final double speedModifier;
|
|
private final PathNavigation navigation;
|
|
private int timeToRecalcPath;
|
|
private final float stopDistance;
|
|
private float oldWaterCost;
|
|
private final float areaSize;
|
|
|
|
public FollowMobGoal(Mob p_25271_, double p_25272_, float p_25273_, float p_25274_) {
|
|
this.mob = p_25271_;
|
|
this.followPredicate = p_25278_ -> p_25278_ != null && p_25271_.getClass() != p_25278_.getClass();
|
|
this.speedModifier = p_25272_;
|
|
this.navigation = p_25271_.getNavigation();
|
|
this.stopDistance = p_25273_;
|
|
this.areaSize = p_25274_;
|
|
this.setFlags(EnumSet.of(Goal.Flag.MOVE, Goal.Flag.LOOK));
|
|
if (!(p_25271_.getNavigation() instanceof GroundPathNavigation) && !(p_25271_.getNavigation() instanceof FlyingPathNavigation)) {
|
|
throw new IllegalArgumentException("Unsupported mob type for FollowMobGoal");
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public boolean canUse() {
|
|
List<Mob> list = this.mob.level().getEntitiesOfClass(Mob.class, this.mob.getBoundingBox().inflate(this.areaSize), this.followPredicate);
|
|
if (!list.isEmpty()) {
|
|
for (Mob mob : list) {
|
|
if (!mob.isInvisible()) {
|
|
this.followingMob = mob;
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
public boolean canContinueToUse() {
|
|
return this.followingMob != null && !this.navigation.isDone() && this.mob.distanceToSqr(this.followingMob) > this.stopDistance * this.stopDistance;
|
|
}
|
|
|
|
@Override
|
|
public void start() {
|
|
this.timeToRecalcPath = 0;
|
|
this.oldWaterCost = this.mob.getPathfindingMalus(PathType.WATER);
|
|
this.mob.setPathfindingMalus(PathType.WATER, 0.0F);
|
|
}
|
|
|
|
@Override
|
|
public void stop() {
|
|
this.followingMob = null;
|
|
this.navigation.stop();
|
|
this.mob.setPathfindingMalus(PathType.WATER, this.oldWaterCost);
|
|
}
|
|
|
|
@Override
|
|
public void tick() {
|
|
if (this.followingMob != null && !this.mob.isLeashed()) {
|
|
this.mob.getLookControl().setLookAt(this.followingMob, 10.0F, this.mob.getMaxHeadXRot());
|
|
if (--this.timeToRecalcPath <= 0) {
|
|
this.timeToRecalcPath = this.adjustedTickDelay(10);
|
|
double d0 = this.mob.getX() - this.followingMob.getX();
|
|
double d1 = this.mob.getY() - this.followingMob.getY();
|
|
double d2 = this.mob.getZ() - this.followingMob.getZ();
|
|
double d3 = d0 * d0 + d1 * d1 + d2 * d2;
|
|
if (!(d3 <= this.stopDistance * this.stopDistance)) {
|
|
this.navigation.moveTo(this.followingMob, this.speedModifier);
|
|
} else {
|
|
this.navigation.stop();
|
|
LookControl lookcontrol = this.followingMob.getLookControl();
|
|
if (d3 <= this.stopDistance
|
|
|| lookcontrol.getWantedX() == this.mob.getX()
|
|
&& lookcontrol.getWantedY() == this.mob.getY()
|
|
&& lookcontrol.getWantedZ() == this.mob.getZ()) {
|
|
double d4 = this.followingMob.getX() - this.mob.getX();
|
|
double d5 = this.followingMob.getZ() - this.mob.getZ();
|
|
this.navigation.moveTo(this.mob.getX() - d4, this.mob.getY(), this.mob.getZ() - d5, this.speedModifier);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} |