211 lines
6.5 KiB
Java
211 lines
6.5 KiB
Java
package net.minecraft.world.entity.animal;
|
|
|
|
import net.minecraft.core.BlockPos;
|
|
import net.minecraft.nbt.CompoundTag;
|
|
import net.minecraft.network.syncher.EntityDataAccessor;
|
|
import net.minecraft.network.syncher.EntityDataSerializers;
|
|
import net.minecraft.network.syncher.SynchedEntityData;
|
|
import net.minecraft.sounds.SoundEvent;
|
|
import net.minecraft.sounds.SoundEvents;
|
|
import net.minecraft.tags.FluidTags;
|
|
import net.minecraft.util.Mth;
|
|
import net.minecraft.world.InteractionHand;
|
|
import net.minecraft.world.InteractionResult;
|
|
import net.minecraft.world.entity.EntitySelector;
|
|
import net.minecraft.world.entity.EntityType;
|
|
import net.minecraft.world.entity.Mob;
|
|
import net.minecraft.world.entity.MoverType;
|
|
import net.minecraft.world.entity.ai.attributes.Attributes;
|
|
import net.minecraft.world.entity.ai.attributes.AttributeSupplier.Builder;
|
|
import net.minecraft.world.entity.ai.control.MoveControl;
|
|
import net.minecraft.world.entity.ai.goal.AvoidEntityGoal;
|
|
import net.minecraft.world.entity.ai.goal.PanicGoal;
|
|
import net.minecraft.world.entity.ai.goal.RandomSwimmingGoal;
|
|
import net.minecraft.world.entity.ai.navigation.PathNavigation;
|
|
import net.minecraft.world.entity.ai.navigation.WaterBoundPathNavigation;
|
|
import net.minecraft.world.entity.player.Player;
|
|
import net.minecraft.world.item.ItemStack;
|
|
import net.minecraft.world.level.Level;
|
|
import net.minecraft.world.level.block.state.BlockState;
|
|
import net.minecraft.world.phys.Vec3;
|
|
|
|
public abstract class AbstractFish extends WaterAnimal implements Bucketable {
|
|
private static final EntityDataAccessor<Boolean> FROM_BUCKET = SynchedEntityData.defineId(AbstractFish.class, EntityDataSerializers.BOOLEAN);
|
|
private static final boolean DEFAULT_FROM_BUCKET = false;
|
|
|
|
public AbstractFish(EntityType<? extends AbstractFish> entityType, Level level) {
|
|
super(entityType, level);
|
|
this.moveControl = new AbstractFish.FishMoveControl(this);
|
|
}
|
|
|
|
public static Builder createAttributes() {
|
|
return Mob.createMobAttributes().add(Attributes.MAX_HEALTH, 3.0);
|
|
}
|
|
|
|
@Override
|
|
public boolean requiresCustomPersistence() {
|
|
return super.requiresCustomPersistence() || this.fromBucket();
|
|
}
|
|
|
|
@Override
|
|
public boolean removeWhenFarAway(double distanceToClosestPlayer) {
|
|
return !this.fromBucket() && !this.hasCustomName();
|
|
}
|
|
|
|
@Override
|
|
public int getMaxSpawnClusterSize() {
|
|
return 8;
|
|
}
|
|
|
|
@Override
|
|
protected void defineSynchedData(net.minecraft.network.syncher.SynchedEntityData.Builder builder) {
|
|
super.defineSynchedData(builder);
|
|
builder.define(FROM_BUCKET, false);
|
|
}
|
|
|
|
@Override
|
|
public boolean fromBucket() {
|
|
return this.entityData.get(FROM_BUCKET);
|
|
}
|
|
|
|
@Override
|
|
public void setFromBucket(boolean fromBucket) {
|
|
this.entityData.set(FROM_BUCKET, fromBucket);
|
|
}
|
|
|
|
@Override
|
|
public void addAdditionalSaveData(CompoundTag tag) {
|
|
super.addAdditionalSaveData(tag);
|
|
tag.putBoolean("FromBucket", this.fromBucket());
|
|
}
|
|
|
|
@Override
|
|
public void readAdditionalSaveData(CompoundTag tag) {
|
|
super.readAdditionalSaveData(tag);
|
|
this.setFromBucket(tag.getBooleanOr("FromBucket", false));
|
|
}
|
|
|
|
@Override
|
|
protected void registerGoals() {
|
|
super.registerGoals();
|
|
this.goalSelector.addGoal(0, new PanicGoal(this, 1.25));
|
|
this.goalSelector.addGoal(2, new AvoidEntityGoal(this, Player.class, 8.0F, 1.6, 1.4, EntitySelector.NO_SPECTATORS::test));
|
|
this.goalSelector.addGoal(4, new AbstractFish.FishSwimGoal(this));
|
|
}
|
|
|
|
@Override
|
|
protected PathNavigation createNavigation(Level level) {
|
|
return new WaterBoundPathNavigation(this, level);
|
|
}
|
|
|
|
@Override
|
|
public void travel(Vec3 travelVector) {
|
|
if (this.isInWater()) {
|
|
this.moveRelative(0.01F, travelVector);
|
|
this.move(MoverType.SELF, this.getDeltaMovement());
|
|
this.setDeltaMovement(this.getDeltaMovement().scale(0.9));
|
|
if (this.getTarget() == null) {
|
|
this.setDeltaMovement(this.getDeltaMovement().add(0.0, -0.005, 0.0));
|
|
}
|
|
} else {
|
|
super.travel(travelVector);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void aiStep() {
|
|
if (!this.isInWater() && this.onGround() && this.verticalCollision) {
|
|
this.setDeltaMovement(this.getDeltaMovement().add((this.random.nextFloat() * 2.0F - 1.0F) * 0.05F, 0.4F, (this.random.nextFloat() * 2.0F - 1.0F) * 0.05F));
|
|
this.setOnGround(false);
|
|
this.hasImpulse = true;
|
|
this.makeSound(this.getFlopSound());
|
|
}
|
|
|
|
super.aiStep();
|
|
}
|
|
|
|
@Override
|
|
protected InteractionResult mobInteract(Player player, InteractionHand hand) {
|
|
return (InteractionResult)Bucketable.bucketMobPickup(player, hand, this).orElse(super.mobInteract(player, hand));
|
|
}
|
|
|
|
@Override
|
|
public void saveToBucketTag(ItemStack stack) {
|
|
Bucketable.saveDefaultDataToBucketTag(this, stack);
|
|
}
|
|
|
|
@Override
|
|
public void loadFromBucketTag(CompoundTag tag) {
|
|
Bucketable.loadDefaultDataFromBucketTag(this, tag);
|
|
}
|
|
|
|
@Override
|
|
public SoundEvent getPickupSound() {
|
|
return SoundEvents.BUCKET_FILL_FISH;
|
|
}
|
|
|
|
protected boolean canRandomSwim() {
|
|
return true;
|
|
}
|
|
|
|
protected abstract SoundEvent getFlopSound();
|
|
|
|
@Override
|
|
protected SoundEvent getSwimSound() {
|
|
return SoundEvents.FISH_SWIM;
|
|
}
|
|
|
|
@Override
|
|
protected void playStepSound(BlockPos pos, BlockState state) {
|
|
}
|
|
|
|
static class FishMoveControl extends MoveControl {
|
|
private final AbstractFish fish;
|
|
|
|
FishMoveControl(AbstractFish fish) {
|
|
super(fish);
|
|
this.fish = fish;
|
|
}
|
|
|
|
@Override
|
|
public void tick() {
|
|
if (this.fish.isEyeInFluid(FluidTags.WATER)) {
|
|
this.fish.setDeltaMovement(this.fish.getDeltaMovement().add(0.0, 0.005, 0.0));
|
|
}
|
|
|
|
if (this.operation == MoveControl.Operation.MOVE_TO && !this.fish.getNavigation().isDone()) {
|
|
float f = (float)(this.speedModifier * this.fish.getAttributeValue(Attributes.MOVEMENT_SPEED));
|
|
this.fish.setSpeed(Mth.lerp(0.125F, this.fish.getSpeed(), f));
|
|
double d = this.wantedX - this.fish.getX();
|
|
double e = this.wantedY - this.fish.getY();
|
|
double g = this.wantedZ - this.fish.getZ();
|
|
if (e != 0.0) {
|
|
double h = Math.sqrt(d * d + e * e + g * g);
|
|
this.fish.setDeltaMovement(this.fish.getDeltaMovement().add(0.0, this.fish.getSpeed() * (e / h) * 0.1, 0.0));
|
|
}
|
|
|
|
if (d != 0.0 || g != 0.0) {
|
|
float i = (float)(Mth.atan2(g, d) * 180.0F / (float)Math.PI) - 90.0F;
|
|
this.fish.setYRot(this.rotlerp(this.fish.getYRot(), i, 90.0F));
|
|
this.fish.yBodyRot = this.fish.getYRot();
|
|
}
|
|
} else {
|
|
this.fish.setSpeed(0.0F);
|
|
}
|
|
}
|
|
}
|
|
|
|
static class FishSwimGoal extends RandomSwimmingGoal {
|
|
private final AbstractFish fish;
|
|
|
|
public FishSwimGoal(AbstractFish fish) {
|
|
super(fish, 1.0, 40);
|
|
this.fish = fish;
|
|
}
|
|
|
|
@Override
|
|
public boolean canUse() {
|
|
return this.fish.canRandomSwim() && super.canUse();
|
|
}
|
|
}
|
|
}
|