minecraft-src/net/minecraft/world/entity/ai/goal/FollowMobGoal.java
2025-07-04 01:41:11 +03:00

104 lines
3.7 KiB
Java

package net.minecraft.world.entity.ai.goal;
import java.util.EnumSet;
import java.util.List;
import java.util.function.Predicate;
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;
import org.jetbrains.annotations.Nullable;
/**
* A goal allowing a mob to follow others. The mob must have Ground or Flying navigation.
*/
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;
/**
* Constructs a goal allowing a mob to follow others. The mob must have Ground or Flying navigation.
*/
public FollowMobGoal(Mob mob, double speedModifier, float stopDistance, float areaSize) {
this.mob = mob;
this.followPredicate = mob2 -> mob2 != null && mob.getClass() != mob2.getClass();
this.speedModifier = speedModifier;
this.navigation = mob.getNavigation();
this.stopDistance = stopDistance;
this.areaSize = areaSize;
this.setFlags(EnumSet.of(Goal.Flag.MOVE, Goal.Flag.LOOK));
if (!(mob.getNavigation() instanceof GroundPathNavigation) && !(mob.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 d = this.mob.getX() - this.followingMob.getX();
double e = this.mob.getY() - this.followingMob.getY();
double f = this.mob.getZ() - this.followingMob.getZ();
double g = d * d + e * e + f * f;
if (!(g <= this.stopDistance * this.stopDistance)) {
this.navigation.moveTo(this.followingMob, this.speedModifier);
} else {
this.navigation.stop();
LookControl lookControl = this.followingMob.getLookControl();
if (g <= this.stopDistance
|| lookControl.getWantedX() == this.mob.getX() && lookControl.getWantedY() == this.mob.getY() && lookControl.getWantedZ() == this.mob.getZ()) {
double h = this.followingMob.getX() - this.mob.getX();
double i = this.followingMob.getZ() - this.mob.getZ();
this.navigation.moveTo(this.mob.getX() - h, this.mob.getY(), this.mob.getZ() - i, this.speedModifier);
}
}
}
}
}
}