package net.minecraft.world.entity.monster; import java.util.EnumSet; import net.minecraft.core.BlockPos; import net.minecraft.core.particles.ParticleTypes; import net.minecraft.network.syncher.EntityDataAccessor; import net.minecraft.network.syncher.EntityDataSerializers; import net.minecraft.network.syncher.SynchedEntityData; import net.minecraft.server.level.ServerLevel; import net.minecraft.sounds.SoundEvent; import net.minecraft.sounds.SoundEvents; import net.minecraft.tags.DamageTypeTags; import net.minecraft.tags.FluidTags; import net.minecraft.util.Mth; import net.minecraft.util.RandomSource; import net.minecraft.world.Difficulty; import net.minecraft.world.damagesource.DamageSource; import net.minecraft.world.damagesource.DamageTypes; import net.minecraft.world.entity.Entity; import net.minecraft.world.entity.EntitySpawnReason; import net.minecraft.world.entity.EntityType; import net.minecraft.world.entity.LivingEntity; 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.LookControl; import net.minecraft.world.entity.ai.control.MoveControl; import net.minecraft.world.entity.ai.goal.Goal; import net.minecraft.world.entity.ai.goal.LookAtPlayerGoal; import net.minecraft.world.entity.ai.goal.MoveTowardsRestrictionGoal; import net.minecraft.world.entity.ai.goal.RandomLookAroundGoal; import net.minecraft.world.entity.ai.goal.RandomStrollGoal; import net.minecraft.world.entity.ai.goal.target.NearestAttackableTargetGoal; import net.minecraft.world.entity.ai.navigation.PathNavigation; import net.minecraft.world.entity.ai.navigation.WaterBoundPathNavigation; import net.minecraft.world.entity.ai.targeting.TargetingConditions; import net.minecraft.world.entity.animal.Squid; import net.minecraft.world.entity.animal.axolotl.Axolotl; import net.minecraft.world.entity.player.Player; import net.minecraft.world.level.Level; import net.minecraft.world.level.LevelAccessor; import net.minecraft.world.level.LevelReader; import net.minecraft.world.level.pathfinder.PathType; import net.minecraft.world.phys.Vec3; import org.jetbrains.annotations.Nullable; public class Guardian extends Monster { protected static final int ATTACK_TIME = 80; private static final EntityDataAccessor DATA_ID_MOVING = SynchedEntityData.defineId(Guardian.class, EntityDataSerializers.BOOLEAN); private static final EntityDataAccessor DATA_ID_ATTACK_TARGET = SynchedEntityData.defineId(Guardian.class, EntityDataSerializers.INT); private float clientSideTailAnimation; private float clientSideTailAnimationO; private float clientSideTailAnimationSpeed; private float clientSideSpikesAnimation; private float clientSideSpikesAnimationO; @Nullable private LivingEntity clientSideCachedAttackTarget; private int clientSideAttackTime; private boolean clientSideTouchedGround; @Nullable protected RandomStrollGoal randomStrollGoal; public Guardian(EntityType entityType, Level level) { super(entityType, level); this.xpReward = 10; this.setPathfindingMalus(PathType.WATER, 0.0F); this.moveControl = new Guardian.GuardianMoveControl(this); this.clientSideTailAnimation = this.random.nextFloat(); this.clientSideTailAnimationO = this.clientSideTailAnimation; } @Override protected void registerGoals() { MoveTowardsRestrictionGoal moveTowardsRestrictionGoal = new MoveTowardsRestrictionGoal(this, 1.0); this.randomStrollGoal = new RandomStrollGoal(this, 1.0, 80); this.goalSelector.addGoal(4, new Guardian.GuardianAttackGoal(this)); this.goalSelector.addGoal(5, moveTowardsRestrictionGoal); this.goalSelector.addGoal(7, this.randomStrollGoal); this.goalSelector.addGoal(8, new LookAtPlayerGoal(this, Player.class, 8.0F)); this.goalSelector.addGoal(8, new LookAtPlayerGoal(this, Guardian.class, 12.0F, 0.01F)); this.goalSelector.addGoal(9, new RandomLookAroundGoal(this)); this.randomStrollGoal.setFlags(EnumSet.of(Goal.Flag.MOVE, Goal.Flag.LOOK)); moveTowardsRestrictionGoal.setFlags(EnumSet.of(Goal.Flag.MOVE, Goal.Flag.LOOK)); this.targetSelector.addGoal(1, new NearestAttackableTargetGoal(this, LivingEntity.class, 10, true, false, new Guardian.GuardianAttackSelector(this))); } public static Builder createAttributes() { return Monster.createMonsterAttributes().add(Attributes.ATTACK_DAMAGE, 6.0).add(Attributes.MOVEMENT_SPEED, 0.5).add(Attributes.MAX_HEALTH, 30.0); } @Override protected PathNavigation createNavigation(Level level) { return new WaterBoundPathNavigation(this, level); } @Override protected void defineSynchedData(net.minecraft.network.syncher.SynchedEntityData.Builder builder) { super.defineSynchedData(builder); builder.define(DATA_ID_MOVING, false); builder.define(DATA_ID_ATTACK_TARGET, 0); } public boolean isMoving() { return this.entityData.get(DATA_ID_MOVING); } void setMoving(boolean moving) { this.entityData.set(DATA_ID_MOVING, moving); } public int getAttackDuration() { return 80; } void setActiveAttackTarget(int activeAttackTargetId) { this.entityData.set(DATA_ID_ATTACK_TARGET, activeAttackTargetId); } public boolean hasActiveAttackTarget() { return this.entityData.get(DATA_ID_ATTACK_TARGET) != 0; } @Nullable public LivingEntity getActiveAttackTarget() { if (!this.hasActiveAttackTarget()) { return null; } else if (this.level().isClientSide) { if (this.clientSideCachedAttackTarget != null) { return this.clientSideCachedAttackTarget; } else { Entity entity = this.level().getEntity(this.entityData.get(DATA_ID_ATTACK_TARGET)); if (entity instanceof LivingEntity) { this.clientSideCachedAttackTarget = (LivingEntity)entity; return this.clientSideCachedAttackTarget; } else { return null; } } } else { return this.getTarget(); } } @Override public void onSyncedDataUpdated(EntityDataAccessor dataAccessor) { super.onSyncedDataUpdated(dataAccessor); if (DATA_ID_ATTACK_TARGET.equals(dataAccessor)) { this.clientSideAttackTime = 0; this.clientSideCachedAttackTarget = null; } } @Override public int getAmbientSoundInterval() { return 160; } @Override protected SoundEvent getAmbientSound() { return this.isInWaterOrBubble() ? SoundEvents.GUARDIAN_AMBIENT : SoundEvents.GUARDIAN_AMBIENT_LAND; } @Override protected SoundEvent getHurtSound(DamageSource damageSource) { return this.isInWaterOrBubble() ? SoundEvents.GUARDIAN_HURT : SoundEvents.GUARDIAN_HURT_LAND; } @Override protected SoundEvent getDeathSound() { return this.isInWaterOrBubble() ? SoundEvents.GUARDIAN_DEATH : SoundEvents.GUARDIAN_DEATH_LAND; } @Override protected Entity.MovementEmission getMovementEmission() { return Entity.MovementEmission.EVENTS; } @Override public float getWalkTargetValue(BlockPos pos, LevelReader level) { return level.getFluidState(pos).is(FluidTags.WATER) ? 10.0F + level.getPathfindingCostFromLightLevels(pos) : super.getWalkTargetValue(pos, level); } @Override public void aiStep() { if (this.isAlive()) { if (this.level().isClientSide) { this.clientSideTailAnimationO = this.clientSideTailAnimation; if (!this.isInWater()) { this.clientSideTailAnimationSpeed = 2.0F; Vec3 vec3 = this.getDeltaMovement(); if (vec3.y > 0.0 && this.clientSideTouchedGround && !this.isSilent()) { this.level().playLocalSound(this.getX(), this.getY(), this.getZ(), this.getFlopSound(), this.getSoundSource(), 1.0F, 1.0F, false); } this.clientSideTouchedGround = vec3.y < 0.0 && this.level().loadedAndEntityCanStandOn(this.blockPosition().below(), this); } else if (this.isMoving()) { if (this.clientSideTailAnimationSpeed < 0.5F) { this.clientSideTailAnimationSpeed = 4.0F; } else { this.clientSideTailAnimationSpeed = this.clientSideTailAnimationSpeed + (0.5F - this.clientSideTailAnimationSpeed) * 0.1F; } } else { this.clientSideTailAnimationSpeed = this.clientSideTailAnimationSpeed + (0.125F - this.clientSideTailAnimationSpeed) * 0.2F; } this.clientSideTailAnimation = this.clientSideTailAnimation + this.clientSideTailAnimationSpeed; this.clientSideSpikesAnimationO = this.clientSideSpikesAnimation; if (!this.isInWaterOrBubble()) { this.clientSideSpikesAnimation = this.random.nextFloat(); } else if (this.isMoving()) { this.clientSideSpikesAnimation = this.clientSideSpikesAnimation + (0.0F - this.clientSideSpikesAnimation) * 0.25F; } else { this.clientSideSpikesAnimation = this.clientSideSpikesAnimation + (1.0F - this.clientSideSpikesAnimation) * 0.06F; } if (this.isMoving() && this.isInWater()) { Vec3 vec3 = this.getViewVector(0.0F); for (int i = 0; i < 2; i++) { this.level() .addParticle( ParticleTypes.BUBBLE, this.getRandomX(0.5) - vec3.x * 1.5, this.getRandomY() - vec3.y * 1.5, this.getRandomZ(0.5) - vec3.z * 1.5, 0.0, 0.0, 0.0 ); } } if (this.hasActiveAttackTarget()) { if (this.clientSideAttackTime < this.getAttackDuration()) { this.clientSideAttackTime++; } LivingEntity livingEntity = this.getActiveAttackTarget(); if (livingEntity != null) { this.getLookControl().setLookAt(livingEntity, 90.0F, 90.0F); this.getLookControl().tick(); double d = this.getAttackAnimationScale(0.0F); double e = livingEntity.getX() - this.getX(); double f = livingEntity.getY(0.5) - this.getEyeY(); double g = livingEntity.getZ() - this.getZ(); double h = Math.sqrt(e * e + f * f + g * g); e /= h; f /= h; g /= h; double j = this.random.nextDouble(); while (j < h) { j += 1.8 - d + this.random.nextDouble() * (1.7 - d); this.level().addParticle(ParticleTypes.BUBBLE, this.getX() + e * j, this.getEyeY() + f * j, this.getZ() + g * j, 0.0, 0.0, 0.0); } } } } if (this.isInWaterOrBubble()) { this.setAirSupply(300); } else if (this.onGround()) { this.setDeltaMovement(this.getDeltaMovement().add((this.random.nextFloat() * 2.0F - 1.0F) * 0.4F, 0.5, (this.random.nextFloat() * 2.0F - 1.0F) * 0.4F)); this.setYRot(this.random.nextFloat() * 360.0F); this.setOnGround(false); this.hasImpulse = true; } if (this.hasActiveAttackTarget()) { this.setYRot(this.yHeadRot); } } super.aiStep(); } protected SoundEvent getFlopSound() { return SoundEvents.GUARDIAN_FLOP; } public float getTailAnimation(float partialTick) { return Mth.lerp(partialTick, this.clientSideTailAnimationO, this.clientSideTailAnimation); } public float getSpikesAnimation(float partialTick) { return Mth.lerp(partialTick, this.clientSideSpikesAnimationO, this.clientSideSpikesAnimation); } public float getAttackAnimationScale(float partialTick) { return (this.clientSideAttackTime + partialTick) / this.getAttackDuration(); } public float getClientSideAttackTime() { return this.clientSideAttackTime; } @Override public boolean checkSpawnObstruction(LevelReader level) { return level.isUnobstructed(this); } public static boolean checkGuardianSpawnRules( EntityType entityType, LevelAccessor levelAccessor, EntitySpawnReason entitySpawnReason, BlockPos blockPos, RandomSource randomSource ) { return (randomSource.nextInt(20) == 0 || !levelAccessor.canSeeSkyFromBelowWater(blockPos)) && levelAccessor.getDifficulty() != Difficulty.PEACEFUL && (EntitySpawnReason.isSpawner(entitySpawnReason) || levelAccessor.getFluidState(blockPos).is(FluidTags.WATER)) && levelAccessor.getFluidState(blockPos.below()).is(FluidTags.WATER); } @Override public boolean hurtServer(ServerLevel serverLevel, DamageSource damageSource, float f) { if (!this.isMoving() && !damageSource.is(DamageTypeTags.AVOIDS_GUARDIAN_THORNS) && !damageSource.is(DamageTypes.THORNS) && damageSource.getDirectEntity() instanceof LivingEntity livingEntity) { livingEntity.hurtServer(serverLevel, this.damageSources().thorns(this), 2.0F); } if (this.randomStrollGoal != null) { this.randomStrollGoal.trigger(); } return super.hurtServer(serverLevel, damageSource, f); } @Override public int getMaxHeadXRot() { return 180; } @Override public void travel(Vec3 travelVector) { if (this.isControlledByLocalInstance() && this.isInWater()) { this.moveRelative(0.1F, travelVector); this.move(MoverType.SELF, this.getDeltaMovement()); this.setDeltaMovement(this.getDeltaMovement().scale(0.9)); if (!this.isMoving() && this.getTarget() == null) { this.setDeltaMovement(this.getDeltaMovement().add(0.0, -0.005, 0.0)); } } else { super.travel(travelVector); } } static class GuardianAttackGoal extends Goal { private final Guardian guardian; private int attackTime; private final boolean elder; public GuardianAttackGoal(Guardian guardian) { this.guardian = guardian; this.elder = guardian instanceof ElderGuardian; this.setFlags(EnumSet.of(Goal.Flag.MOVE, Goal.Flag.LOOK)); } @Override public boolean canUse() { LivingEntity livingEntity = this.guardian.getTarget(); return livingEntity != null && livingEntity.isAlive(); } @Override public boolean canContinueToUse() { return super.canContinueToUse() && (this.elder || this.guardian.getTarget() != null && this.guardian.distanceToSqr(this.guardian.getTarget()) > 9.0); } @Override public void start() { this.attackTime = -10; this.guardian.getNavigation().stop(); LivingEntity livingEntity = this.guardian.getTarget(); if (livingEntity != null) { this.guardian.getLookControl().setLookAt(livingEntity, 90.0F, 90.0F); } this.guardian.hasImpulse = true; } @Override public void stop() { this.guardian.setActiveAttackTarget(0); this.guardian.setTarget(null); this.guardian.randomStrollGoal.trigger(); } @Override public boolean requiresUpdateEveryTick() { return true; } @Override public void tick() { LivingEntity livingEntity = this.guardian.getTarget(); if (livingEntity != null) { this.guardian.getNavigation().stop(); this.guardian.getLookControl().setLookAt(livingEntity, 90.0F, 90.0F); if (!this.guardian.hasLineOfSight(livingEntity)) { this.guardian.setTarget(null); } else { this.attackTime++; if (this.attackTime == 0) { this.guardian.setActiveAttackTarget(livingEntity.getId()); if (!this.guardian.isSilent()) { this.guardian.level().broadcastEntityEvent(this.guardian, (byte)21); } } else if (this.attackTime >= this.guardian.getAttackDuration()) { float f = 1.0F; if (this.guardian.level().getDifficulty() == Difficulty.HARD) { f += 2.0F; } if (this.elder) { f += 2.0F; } ServerLevel serverLevel = getServerLevel(this.guardian); livingEntity.hurtServer(serverLevel, this.guardian.damageSources().indirectMagic(this.guardian, this.guardian), f); this.guardian.doHurtTarget(serverLevel, livingEntity); this.guardian.setTarget(null); } super.tick(); } } } } static class GuardianAttackSelector implements TargetingConditions.Selector { private final Guardian guardian; public GuardianAttackSelector(Guardian guardian) { this.guardian = guardian; } @Override public boolean test(@Nullable LivingEntity livingEntity, ServerLevel serverLevel) { return (livingEntity instanceof Player || livingEntity instanceof Squid || livingEntity instanceof Axolotl) && livingEntity.distanceToSqr(this.guardian) > 9.0; } } static class GuardianMoveControl extends MoveControl { private final Guardian guardian; public GuardianMoveControl(Guardian guardian) { super(guardian); this.guardian = guardian; } @Override public void tick() { if (this.operation == MoveControl.Operation.MOVE_TO && !this.guardian.getNavigation().isDone()) { Vec3 vec3 = new Vec3(this.wantedX - this.guardian.getX(), this.wantedY - this.guardian.getY(), this.wantedZ - this.guardian.getZ()); double d = vec3.length(); double e = vec3.x / d; double f = vec3.y / d; double g = vec3.z / d; float h = (float)(Mth.atan2(vec3.z, vec3.x) * 180.0F / (float)Math.PI) - 90.0F; this.guardian.setYRot(this.rotlerp(this.guardian.getYRot(), h, 90.0F)); this.guardian.yBodyRot = this.guardian.getYRot(); float i = (float)(this.speedModifier * this.guardian.getAttributeValue(Attributes.MOVEMENT_SPEED)); float j = Mth.lerp(0.125F, this.guardian.getSpeed(), i); this.guardian.setSpeed(j); double k = Math.sin((this.guardian.tickCount + this.guardian.getId()) * 0.5) * 0.05; double l = Math.cos(this.guardian.getYRot() * (float) (Math.PI / 180.0)); double m = Math.sin(this.guardian.getYRot() * (float) (Math.PI / 180.0)); double n = Math.sin((this.guardian.tickCount + this.guardian.getId()) * 0.75) * 0.05; this.guardian.setDeltaMovement(this.guardian.getDeltaMovement().add(k * l, n * (m + l) * 0.25 + j * f * 0.1, k * m)); LookControl lookControl = this.guardian.getLookControl(); double o = this.guardian.getX() + e * 2.0; double p = this.guardian.getEyeY() + f / d; double q = this.guardian.getZ() + g * 2.0; double r = lookControl.getWantedX(); double s = lookControl.getWantedY(); double t = lookControl.getWantedZ(); if (!lookControl.isLookingAtTarget()) { r = o; s = p; t = q; } this.guardian.getLookControl().setLookAt(Mth.lerp(0.125, r, o), Mth.lerp(0.125, s, p), Mth.lerp(0.125, t, q), 10.0F, 40.0F); this.guardian.setMoving(true); } else { this.guardian.setSpeed(0.0F); this.guardian.setMoving(false); } } } }