package net.minecraft.world.entity; import com.google.common.collect.Lists; import com.google.common.collect.Maps; import com.mojang.logging.LogUtils; import java.util.List; import java.util.Map; import java.util.UUID; import net.minecraft.core.Holder; import net.minecraft.core.particles.ColorParticleOption; import net.minecraft.core.particles.ParticleOptions; import net.minecraft.core.particles.ParticleTypes; import net.minecraft.nbt.CompoundTag; import net.minecraft.nbt.NbtOps; import net.minecraft.nbt.Tag; import net.minecraft.network.syncher.EntityDataAccessor; import net.minecraft.network.syncher.EntityDataSerializers; import net.minecraft.network.syncher.SynchedEntityData; import net.minecraft.resources.RegistryOps; import net.minecraft.server.level.ServerLevel; import net.minecraft.util.FastColor; import net.minecraft.util.Mth; import net.minecraft.world.effect.MobEffectInstance; import net.minecraft.world.item.alchemy.Potion; import net.minecraft.world.item.alchemy.PotionContents; import net.minecraft.world.level.Level; import net.minecraft.world.level.material.PushReaction; import org.jetbrains.annotations.Nullable; import org.slf4j.Logger; public class AreaEffectCloud extends Entity implements TraceableEntity { private static final Logger LOGGER = LogUtils.getLogger(); private static final int TIME_BETWEEN_APPLICATIONS = 5; private static final EntityDataAccessor DATA_RADIUS = SynchedEntityData.defineId(AreaEffectCloud.class, EntityDataSerializers.FLOAT); private static final EntityDataAccessor DATA_WAITING = SynchedEntityData.defineId(AreaEffectCloud.class, EntityDataSerializers.BOOLEAN); private static final EntityDataAccessor DATA_PARTICLE = SynchedEntityData.defineId(AreaEffectCloud.class, EntityDataSerializers.PARTICLE); private static final float MAX_RADIUS = 32.0F; private static final float MINIMAL_RADIUS = 0.5F; private static final float DEFAULT_RADIUS = 3.0F; public static final float DEFAULT_WIDTH = 6.0F; public static final float HEIGHT = 0.5F; private PotionContents potionContents = PotionContents.EMPTY; private final Map victims = Maps.newHashMap(); private int duration = 600; private int waitTime = 20; private int reapplicationDelay = 20; private int durationOnUse; private float radiusOnUse; private float radiusPerTick; @Nullable private LivingEntity owner; @Nullable private UUID ownerUUID; public AreaEffectCloud(EntityType entityType, Level level) { super(entityType, level); this.noPhysics = true; } public AreaEffectCloud(Level level, double x, double y, double z) { this(EntityType.AREA_EFFECT_CLOUD, level); this.setPos(x, y, z); } @Override protected void defineSynchedData(SynchedEntityData.Builder builder) { builder.define(DATA_RADIUS, 3.0F); builder.define(DATA_WAITING, false); builder.define(DATA_PARTICLE, ColorParticleOption.create(ParticleTypes.ENTITY_EFFECT, -1)); } public void setRadius(float radius) { if (!this.level().isClientSide) { this.getEntityData().set(DATA_RADIUS, Mth.clamp(radius, 0.0F, 32.0F)); } } @Override public void refreshDimensions() { double d = this.getX(); double e = this.getY(); double f = this.getZ(); super.refreshDimensions(); this.setPos(d, e, f); } public float getRadius() { return this.getEntityData().get(DATA_RADIUS); } public void setPotionContents(PotionContents potionContents) { this.potionContents = potionContents; this.updateColor(); } private void updateColor() { ParticleOptions particleOptions = this.entityData.get(DATA_PARTICLE); if (particleOptions instanceof ColorParticleOption colorParticleOption) { int i = this.potionContents.equals(PotionContents.EMPTY) ? 0 : this.potionContents.getColor(); this.entityData.set(DATA_PARTICLE, ColorParticleOption.create(colorParticleOption.getType(), FastColor.ARGB32.opaque(i))); } } public void addEffect(MobEffectInstance effectInstance) { this.setPotionContents(this.potionContents.withEffectAdded(effectInstance)); } public ParticleOptions getParticle() { return this.getEntityData().get(DATA_PARTICLE); } public void setParticle(ParticleOptions particleOption) { this.getEntityData().set(DATA_PARTICLE, particleOption); } /** * Sets if the cloud is waiting. While waiting, the radius is ignored and the cloud shows fewer particles in its area. */ protected void setWaiting(boolean waiting) { this.getEntityData().set(DATA_WAITING, waiting); } /** * Returns {@code true} if the cloud is waiting. While waiting, the radius is ignored and the cloud shows fewer particles in its area. */ public boolean isWaiting() { return this.getEntityData().get(DATA_WAITING); } public int getDuration() { return this.duration; } public void setDuration(int duration) { this.duration = duration; } @Override public void tick() { super.tick(); boolean bl = this.isWaiting(); float f = this.getRadius(); if (this.level().isClientSide) { if (bl && this.random.nextBoolean()) { return; } ParticleOptions particleOptions = this.getParticle(); int i; float g; if (bl) { i = 2; g = 0.2F; } else { i = Mth.ceil((float) Math.PI * f * f); g = f; } for (int j = 0; j < i; j++) { float h = this.random.nextFloat() * (float) (Math.PI * 2); float k = Mth.sqrt(this.random.nextFloat()) * g; double d = this.getX() + Mth.cos(h) * k; double e = this.getY(); double l = this.getZ() + Mth.sin(h) * k; if (particleOptions.getType() == ParticleTypes.ENTITY_EFFECT) { if (bl && this.random.nextBoolean()) { this.level().addAlwaysVisibleParticle(ColorParticleOption.create(ParticleTypes.ENTITY_EFFECT, -1), d, e, l, 0.0, 0.0, 0.0); } else { this.level().addAlwaysVisibleParticle(particleOptions, d, e, l, 0.0, 0.0, 0.0); } } else if (bl) { this.level().addAlwaysVisibleParticle(particleOptions, d, e, l, 0.0, 0.0, 0.0); } else { this.level().addAlwaysVisibleParticle(particleOptions, d, e, l, (0.5 - this.random.nextDouble()) * 0.15, 0.01F, (0.5 - this.random.nextDouble()) * 0.15); } } } else { if (this.tickCount >= this.waitTime + this.duration) { this.discard(); return; } boolean bl2 = this.tickCount < this.waitTime; if (bl != bl2) { this.setWaiting(bl2); } if (bl2) { return; } if (this.radiusPerTick != 0.0F) { f += this.radiusPerTick; if (f < 0.5F) { this.discard(); return; } this.setRadius(f); } if (this.tickCount % 5 == 0) { this.victims.entrySet().removeIf(entry -> this.tickCount >= (Integer)entry.getValue()); if (!this.potionContents.hasEffects()) { this.victims.clear(); } else { List list = Lists.newArrayList(); if (this.potionContents.potion().isPresent()) { for (MobEffectInstance mobEffectInstance : ((Potion)((Holder)this.potionContents.potion().get()).value()).getEffects()) { list.add( new MobEffectInstance( mobEffectInstance.getEffect(), mobEffectInstance.mapDuration(i -> i / 4), mobEffectInstance.getAmplifier(), mobEffectInstance.isAmbient(), mobEffectInstance.isVisible() ) ); } } list.addAll(this.potionContents.customEffects()); List list2 = this.level().getEntitiesOfClass(LivingEntity.class, this.getBoundingBox()); if (!list2.isEmpty()) { for (LivingEntity livingEntity : list2) { if (!this.victims.containsKey(livingEntity) && livingEntity.isAffectedByPotions() && !list.stream().noneMatch(livingEntity::canBeAffected)) { double m = livingEntity.getX() - this.getX(); double n = livingEntity.getZ() - this.getZ(); double o = m * m + n * n; if (o <= f * f) { this.victims.put(livingEntity, this.tickCount + this.reapplicationDelay); for (MobEffectInstance mobEffectInstance2 : list) { if (mobEffectInstance2.getEffect().value().isInstantenous()) { mobEffectInstance2.getEffect().value().applyInstantenousEffect(this, this.getOwner(), livingEntity, mobEffectInstance2.getAmplifier(), 0.5); } else { livingEntity.addEffect(new MobEffectInstance(mobEffectInstance2), this); } } if (this.radiusOnUse != 0.0F) { f += this.radiusOnUse; if (f < 0.5F) { this.discard(); return; } this.setRadius(f); } if (this.durationOnUse != 0) { this.duration = this.duration + this.durationOnUse; if (this.duration <= 0) { this.discard(); return; } } } } } } } } } } public float getRadiusOnUse() { return this.radiusOnUse; } public void setRadiusOnUse(float radiusOnUse) { this.radiusOnUse = radiusOnUse; } public float getRadiusPerTick() { return this.radiusPerTick; } public void setRadiusPerTick(float radiusPerTick) { this.radiusPerTick = radiusPerTick; } public int getDurationOnUse() { return this.durationOnUse; } public void setDurationOnUse(int durationOnUse) { this.durationOnUse = durationOnUse; } public int getWaitTime() { return this.waitTime; } public void setWaitTime(int waitTime) { this.waitTime = waitTime; } public void setOwner(@Nullable LivingEntity owner) { this.owner = owner; this.ownerUUID = owner == null ? null : owner.getUUID(); } @Nullable public LivingEntity getOwner() { if (this.owner == null && this.ownerUUID != null && this.level() instanceof ServerLevel) { Entity entity = ((ServerLevel)this.level()).getEntity(this.ownerUUID); if (entity instanceof LivingEntity) { this.owner = (LivingEntity)entity; } } return this.owner; } @Override protected void readAdditionalSaveData(CompoundTag compound) { this.tickCount = compound.getInt("Age"); this.duration = compound.getInt("Duration"); this.waitTime = compound.getInt("WaitTime"); this.reapplicationDelay = compound.getInt("ReapplicationDelay"); this.durationOnUse = compound.getInt("DurationOnUse"); this.radiusOnUse = compound.getFloat("RadiusOnUse"); this.radiusPerTick = compound.getFloat("RadiusPerTick"); this.setRadius(compound.getFloat("Radius")); if (compound.hasUUID("Owner")) { this.ownerUUID = compound.getUUID("Owner"); } RegistryOps registryOps = this.registryAccess().createSerializationContext(NbtOps.INSTANCE); if (compound.contains("Particle", 10)) { ParticleTypes.CODEC .parse(registryOps, compound.get("Particle")) .resultOrPartial(string -> LOGGER.warn("Failed to parse area effect cloud particle options: '{}'", string)) .ifPresent(this::setParticle); } if (compound.contains("potion_contents")) { PotionContents.CODEC .parse(registryOps, compound.get("potion_contents")) .resultOrPartial(string -> LOGGER.warn("Failed to parse area effect cloud potions: '{}'", string)) .ifPresent(this::setPotionContents); } } @Override protected void addAdditionalSaveData(CompoundTag compound) { compound.putInt("Age", this.tickCount); compound.putInt("Duration", this.duration); compound.putInt("WaitTime", this.waitTime); compound.putInt("ReapplicationDelay", this.reapplicationDelay); compound.putInt("DurationOnUse", this.durationOnUse); compound.putFloat("RadiusOnUse", this.radiusOnUse); compound.putFloat("RadiusPerTick", this.radiusPerTick); compound.putFloat("Radius", this.getRadius()); RegistryOps registryOps = this.registryAccess().createSerializationContext(NbtOps.INSTANCE); compound.put("Particle", ParticleTypes.CODEC.encodeStart(registryOps, this.getParticle()).getOrThrow()); if (this.ownerUUID != null) { compound.putUUID("Owner", this.ownerUUID); } if (!this.potionContents.equals(PotionContents.EMPTY)) { Tag tag = PotionContents.CODEC.encodeStart(registryOps, this.potionContents).getOrThrow(); compound.put("potion_contents", tag); } } @Override public void onSyncedDataUpdated(EntityDataAccessor dataAccessor) { if (DATA_RADIUS.equals(dataAccessor)) { this.refreshDimensions(); } super.onSyncedDataUpdated(dataAccessor); } @Override public PushReaction getPistonPushReaction() { return PushReaction.IGNORE; } @Override public EntityDimensions getDimensions(Pose pose) { return EntityDimensions.scalable(this.getRadius() * 2.0F, 0.5F); } }