350 lines
14 KiB
Java
350 lines
14 KiB
Java
package com.atsuishio.superbwarfare.entity.projectile;
|
|
|
|
import com.atsuishio.superbwarfare.config.server.ExplosionConfig;
|
|
import com.atsuishio.superbwarfare.entity.LoudlyEntity;
|
|
import com.atsuishio.superbwarfare.entity.vehicle.base.VehicleEntity;
|
|
import com.atsuishio.superbwarfare.init.ModDamageTypes;
|
|
import com.atsuishio.superbwarfare.init.ModEntities;
|
|
import com.atsuishio.superbwarfare.init.ModItems;
|
|
import com.atsuishio.superbwarfare.init.ModSounds;
|
|
import com.atsuishio.superbwarfare.network.message.receive.ClientIndicatorMessage;
|
|
import com.atsuishio.superbwarfare.tools.*;
|
|
import net.minecraft.core.BlockPos;
|
|
import net.minecraft.core.particles.ParticleTypes;
|
|
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.server.level.ServerLevel;
|
|
import net.minecraft.server.level.ServerPlayer;
|
|
import net.minecraft.sounds.SoundEvent;
|
|
import net.minecraft.sounds.SoundEvents;
|
|
import net.minecraft.sounds.SoundSource;
|
|
import net.minecraft.util.Mth;
|
|
import net.minecraft.world.damagesource.DamageSource;
|
|
import net.minecraft.world.damagesource.DamageTypes;
|
|
import net.minecraft.world.entity.AreaEffectCloud;
|
|
import net.minecraft.world.entity.Entity;
|
|
import net.minecraft.world.entity.EntityType;
|
|
import net.minecraft.world.entity.LivingEntity;
|
|
import net.minecraft.world.entity.animal.Pig;
|
|
import net.minecraft.world.entity.boss.enderdragon.EnderDragon;
|
|
import net.minecraft.world.entity.projectile.ThrownPotion;
|
|
import net.minecraft.world.item.Item;
|
|
import net.minecraft.world.level.Explosion;
|
|
import net.minecraft.world.level.Level;
|
|
import net.minecraft.world.level.block.Block;
|
|
import net.minecraft.world.phys.BlockHitResult;
|
|
import net.minecraft.world.phys.EntityHitResult;
|
|
import net.minecraft.world.phys.HitResult;
|
|
import net.minecraft.world.phys.Vec3;
|
|
import net.neoforged.neoforge.event.EventHooks;
|
|
import net.neoforged.neoforge.network.PacketDistributor;
|
|
import org.jetbrains.annotations.NotNull;
|
|
import software.bernie.geckolib.animatable.GeoEntity;
|
|
import software.bernie.geckolib.animatable.instance.AnimatableInstanceCache;
|
|
import software.bernie.geckolib.animation.*;
|
|
import software.bernie.geckolib.util.GeckoLibUtil;
|
|
|
|
import java.util.List;
|
|
|
|
public class Agm65Entity extends FastThrowableProjectile implements GeoEntity, DestroyableProjectileEntity, LoudlyEntity, ExplosiveProjectile {
|
|
public static final EntityDataAccessor<Float> HEALTH = SynchedEntityData.defineId(Agm65Entity.class, EntityDataSerializers.FLOAT);
|
|
public static final EntityDataAccessor<String> TARGET_UUID = SynchedEntityData.defineId(Agm65Entity.class, EntityDataSerializers.STRING);
|
|
private final AnimatableInstanceCache cache = GeckoLibUtil.createInstanceCache(this);
|
|
|
|
private float damage = ExplosionConfig.AGM_65_DAMAGE.get();
|
|
private float explosionDamage = ExplosionConfig.AGM_65_EXPLOSION_DAMAGE.get();
|
|
private float explosionRadius = ExplosionConfig.AGM_65_EXPLOSION_RADIUS.get().floatValue();
|
|
private boolean distracted = false;
|
|
private int durability = 40;
|
|
private boolean firstHit = true;
|
|
|
|
public Agm65Entity(EntityType<? extends Agm65Entity> type, Level world) {
|
|
super(type, world);
|
|
this.noCulling = true;
|
|
}
|
|
|
|
public Agm65Entity(LivingEntity entity, Level level) {
|
|
super(ModEntities.AGM_65.get(), entity, level);
|
|
}
|
|
|
|
public void setTargetUuid(String uuid) {
|
|
this.entityData.set(TARGET_UUID, uuid);
|
|
}
|
|
|
|
@Override
|
|
protected @NotNull Item getDefaultItem() {
|
|
return ModItems.AGM.get();
|
|
}
|
|
|
|
@Override
|
|
public boolean hurt(DamageSource source, float amount) {
|
|
if (source.getDirectEntity() instanceof ThrownPotion || source.getDirectEntity() instanceof AreaEffectCloud)
|
|
return false;
|
|
if (source.is(DamageTypes.FALL))
|
|
return false;
|
|
if (source.is(DamageTypes.CACTUS))
|
|
return false;
|
|
if (source.is(DamageTypes.DROWN))
|
|
return false;
|
|
if (source.is(DamageTypes.DRAGON_BREATH))
|
|
return false;
|
|
if (source.is(DamageTypes.WITHER))
|
|
return false;
|
|
if (source.is(DamageTypes.WITHER_SKULL))
|
|
return false;
|
|
this.entityData.set(HEALTH, this.entityData.get(HEALTH) - amount);
|
|
|
|
return super.hurt(source, amount);
|
|
}
|
|
|
|
@Override
|
|
protected void defineSynchedData(SynchedEntityData.@NotNull Builder builder) {
|
|
super.defineSynchedData(builder);
|
|
|
|
builder.define(HEALTH, 30f)
|
|
.define(TARGET_UUID, "none");
|
|
}
|
|
|
|
@Override
|
|
public boolean isPickable() {
|
|
return !this.isRemoved();
|
|
}
|
|
|
|
@Override
|
|
public void readAdditionalSaveData(@NotNull CompoundTag compound) {
|
|
super.readAdditionalSaveData(compound);
|
|
if (compound.contains("Health")) {
|
|
this.entityData.set(HEALTH, compound.getFloat("Health"));
|
|
}
|
|
if (compound.contains("Damage")) {
|
|
this.damage = compound.getFloat("Damage");
|
|
}
|
|
if (compound.contains("ExplosionDamage")) {
|
|
this.explosionDamage = compound.getFloat("ExplosionDamage");
|
|
}
|
|
if (compound.contains("Radius")) {
|
|
this.explosionRadius = compound.getFloat("Radius");
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void addAdditionalSaveData(@NotNull CompoundTag compound) {
|
|
super.addAdditionalSaveData(compound);
|
|
compound.putFloat("Health", this.entityData.get(HEALTH));
|
|
compound.putFloat("Damage", this.damage);
|
|
compound.putFloat("ExplosionDamage", this.explosionDamage);
|
|
compound.putFloat("Radius", this.explosionRadius);
|
|
}
|
|
|
|
@Override
|
|
public boolean shouldRenderAtSqrDistance(double pDistance) {
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
protected void onHitEntity(EntityHitResult result) {
|
|
Entity entity = result.getEntity();
|
|
if (this.level() instanceof ServerLevel && tickCount > 8) {
|
|
if (entity == this.getOwner() || (this.getOwner() != null && entity == this.getOwner().getVehicle()))
|
|
return;
|
|
if (this.getOwner() instanceof LivingEntity living) {
|
|
if (!living.level().isClientSide() && living instanceof ServerPlayer player) {
|
|
living.level().playSound(null, living.blockPosition(), ModSounds.INDICATION.get(), SoundSource.VOICE, 1, 1);
|
|
|
|
PacketDistributor.sendToPlayer(player, new ClientIndicatorMessage(0, 5));
|
|
}
|
|
}
|
|
|
|
entity.hurt(ModDamageTypes.causeCannonFireDamage(this.level().registryAccess(), this, this.getOwner()), this.damage);
|
|
|
|
if (entity instanceof LivingEntity) {
|
|
entity.invulnerableTime = 0;
|
|
}
|
|
causeExplode(result);
|
|
this.discard();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public boolean isNoGravity() {
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public void onHitBlock(@NotNull BlockHitResult blockHitResult) {
|
|
if (this.level() instanceof ServerLevel) {
|
|
double x = blockHitResult.getLocation().x;
|
|
double y = blockHitResult.getLocation().y;
|
|
double z = blockHitResult.getLocation().z;
|
|
|
|
if (ExplosionConfig.EXPLOSION_DESTROY.get()) {
|
|
float hardness = this.level().getBlockState(BlockPos.containing(x, y, z)).getBlock().defaultDestroyTime();
|
|
if (hardness <= 50) {
|
|
BlockPos blockPos = BlockPos.containing(x, y, z);
|
|
Block.dropResources(this.level().getBlockState(blockPos), this.level(), BlockPos.containing(x, y, z), null);
|
|
this.level().destroyBlock(blockPos, true);
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < 5; i++) {
|
|
apExplode(blockHitResult, i);
|
|
}
|
|
|
|
causeExplode(blockHitResult);
|
|
}
|
|
}
|
|
|
|
private void apExplode(HitResult result, int index) {
|
|
CustomExplosion explosion = new CustomExplosion(this.level(), this,
|
|
ModDamageTypes.causeProjectileBoomDamage(this.level().registryAccess(),
|
|
this,
|
|
this.getOwner()),
|
|
explosionDamage,
|
|
result.getLocation().x + index * getDeltaMovement().normalize().x,
|
|
result.getLocation().y + index * getDeltaMovement().normalize().y,
|
|
result.getLocation().z + index * getDeltaMovement().normalize().z,
|
|
0.5f * explosionRadius,
|
|
ExplosionConfig.EXPLOSION_DESTROY.get() ? Explosion.BlockInteraction.DESTROY : Explosion.BlockInteraction.KEEP, true).
|
|
setDamageMultiplier(1);
|
|
explosion.explode();
|
|
EventHooks.onExplosionStart(this.level(), explosion);
|
|
explosion.finalizeExplosion(false);
|
|
}
|
|
|
|
@Override
|
|
public void tick() {
|
|
super.tick();
|
|
Entity entity = EntityFindUtil.findEntity(this.level(), entityData.get(TARGET_UUID));
|
|
List<Entity> decoy = SeekTool.seekLivingEntities(this, this.level(), 32, 90);
|
|
|
|
for (var e : decoy) {
|
|
if (e instanceof DecoyEntity decoyEntity && !distracted) {
|
|
this.entityData.set(TARGET_UUID, decoyEntity.getDecoyUUID());
|
|
distracted = true;
|
|
}
|
|
}
|
|
|
|
if (!entityData.get(TARGET_UUID).equals("none")) {
|
|
if (entity != null) {
|
|
if (entity.level() instanceof ServerLevel) {
|
|
if ((!entity.getPassengers().isEmpty() || entity instanceof VehicleEntity) && entity.tickCount % ((int) Math.max(0.04 * this.distanceTo(entity), 2)) == 0) {
|
|
entity.level().playSound(null, entity.getOnPos(), entity instanceof Pig ? SoundEvents.PIG_HURT : ModSounds.MISSILE_WARNING.get(), SoundSource.PLAYERS, 2, 1f);
|
|
}
|
|
|
|
Vec3 targetPos = new Vec3(entity.getX(), entity.getY() + (entity instanceof EnderDragon ? -3 : 0) + 0.15 * distanceTo(entity), entity.getZ());
|
|
|
|
Vec3 toVec = getEyePosition().vectorTo(targetPos).normalize();
|
|
if (this.tickCount > 8) {
|
|
boolean lostTarget = (VectorTool.calculateAngle(getDeltaMovement(), toVec) > 80);
|
|
if (!lostTarget) {
|
|
setDeltaMovement(getDeltaMovement().add(toVec.scale(1.4)).scale(0.75).add(entity.getDeltaMovement()));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (this.tickCount == 8) {
|
|
this.level().playSound(null, BlockPos.containing(position()), ModSounds.MISSILE_START.get(), SoundSource.PLAYERS, 4, 1);
|
|
if (!this.level().isClientSide() && this.level() instanceof ServerLevel serverLevel) {
|
|
ParticleTool.sendParticle(serverLevel, ParticleTypes.CLOUD, this.xo, this.yo, this.zo, 15, 0.8, 0.8, 0.8, 0.01, true);
|
|
ParticleTool.sendParticle(serverLevel, ParticleTypes.CAMPFIRE_COSY_SMOKE, this.xo, this.yo, this.zo, 10, 0.8, 0.8, 0.8, 0.01, true);
|
|
}
|
|
}
|
|
|
|
if (this.tickCount > 8) {
|
|
if (!this.level().isClientSide() && this.level() instanceof ServerLevel serverLevel) {
|
|
ParticleTool.sendParticle(serverLevel, ParticleTypes.CAMPFIRE_COSY_SMOKE, this.xo, this.yo, this.zo, 1, 0, 0, 0, 0, true);
|
|
}
|
|
this.setDeltaMovement(this.getDeltaMovement().multiply(1.06, 1.06, 1.06));
|
|
}
|
|
|
|
if (this.tickCount > 200 || this.isInWater() || this.entityData.get(HEALTH) <= 0) {
|
|
if (this.level() instanceof ServerLevel) {
|
|
ProjectileTool.causeCustomExplode(this,
|
|
ModDamageTypes.causeProjectileBoomDamage(this.level().registryAccess(), this, this.getOwner()),
|
|
this, this.explosionDamage, this.explosionRadius, 1);
|
|
}
|
|
this.discard();
|
|
}
|
|
|
|
float f = (float) Mth.clamp(1 - 0.005 * getDeltaMovement().length(), 0.001, 1);
|
|
|
|
this.setDeltaMovement(this.getDeltaMovement().multiply(f, f, f));
|
|
}
|
|
|
|
private void causeExplode(HitResult result) {
|
|
CustomExplosion explosion = new CustomExplosion(this.level(), this,
|
|
ModDamageTypes.causeProjectileBoomDamage(this.level().registryAccess(),
|
|
this,
|
|
this.getOwner()),
|
|
this.explosionDamage,
|
|
this.getX(),
|
|
this.getEyeY(),
|
|
this.getZ(),
|
|
this.explosionRadius,
|
|
ExplosionConfig.EXPLOSION_DESTROY.get() ? Explosion.BlockInteraction.DESTROY : Explosion.BlockInteraction.KEEP, true).
|
|
setDamageMultiplier(1);
|
|
explosion.explode();
|
|
EventHooks.onExplosionStart(this.level(), explosion);
|
|
explosion.finalizeExplosion(false);
|
|
ParticleTool.spawnHugeExplosionParticles(this.level(), result.getLocation());
|
|
}
|
|
|
|
private PlayState movementPredicate(AnimationState<Agm65Entity> event) {
|
|
return event.setAndContinue(RawAnimation.begin().thenLoop("animation.jvm.idle"));
|
|
}
|
|
|
|
@Override
|
|
protected double getDefaultGravity() {
|
|
return tickCount > 8 ? 0 : 0.15F;
|
|
}
|
|
|
|
@Override
|
|
public void registerControllers(AnimatableManager.ControllerRegistrar data) {
|
|
data.add(new AnimationController<>(this, "movement", 0, this::movementPredicate));
|
|
}
|
|
|
|
@Override
|
|
public AnimatableInstanceCache getAnimatableInstanceCache() {
|
|
return this.cache;
|
|
}
|
|
|
|
@Override
|
|
public boolean shouldSyncMotion() {
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public @NotNull SoundEvent getCloseSound() {
|
|
return ModSounds.ROCKET_ENGINE.get();
|
|
}
|
|
|
|
@Override
|
|
public @NotNull SoundEvent getSound() {
|
|
return ModSounds.ROCKET_FLY.get();
|
|
}
|
|
|
|
@Override
|
|
public float getVolume() {
|
|
return 0.7f;
|
|
}
|
|
|
|
// TODO setter
|
|
@Override
|
|
public void setDamage(float damage) {
|
|
this.damage = damage;
|
|
}
|
|
|
|
@Override
|
|
public void setExplosionDamage(float explosionDamage) {
|
|
this.explosionDamage = explosionDamage;
|
|
}
|
|
|
|
@Override
|
|
public void setExplosionRadius(float radius) {
|
|
this.explosionRadius = radius;
|
|
}
|
|
}
|