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 FROM_BUCKET = SynchedEntityData.defineId(AbstractFish.class, EntityDataSerializers.BOOLEAN); private static final boolean DEFAULT_FROM_BUCKET = false; public AbstractFish(EntityType 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(); } } }