package com.atsuishio.superbwarfare.entity; import com.atsuishio.superbwarfare.config.server.ExplosionConfig; import com.atsuishio.superbwarfare.entity.vehicle.damage.DamageModifier; import com.atsuishio.superbwarfare.init.ModDamageTypes; import com.atsuishio.superbwarfare.init.ModEntities; import com.atsuishio.superbwarfare.init.ModItems; import com.atsuishio.superbwarfare.tools.CustomExplosion; import com.atsuishio.superbwarfare.tools.ParticleTool; 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.players.OldUsersConverter; import net.minecraft.world.InteractionHand; import net.minecraft.world.InteractionResult; import net.minecraft.world.damagesource.DamageSource; import net.minecraft.world.damagesource.DamageTypes; import net.minecraft.world.entity.*; import net.minecraft.world.entity.decoration.HangingEntity; import net.minecraft.world.entity.player.Player; import net.minecraft.world.item.ItemStack; import net.minecraft.world.level.Explosion; import net.minecraft.world.level.Level; import net.minecraft.world.level.entity.EntityTypeTest; import net.minecraft.world.phys.AABB; import net.minecraft.world.phys.Vec3; import net.neoforged.neoforge.event.EventHooks; import net.neoforged.neoforge.items.ItemHandlerHelper; import org.jetbrains.annotations.NotNull; import software.bernie.geckolib.animatable.GeoEntity; import software.bernie.geckolib.animatable.instance.AnimatableInstanceCache; import software.bernie.geckolib.animation.AnimatableManager; import software.bernie.geckolib.util.GeckoLibUtil; import javax.annotation.Nullable; import javax.annotation.ParametersAreNonnullByDefault; import java.util.Optional; import java.util.UUID; public class Tm62Entity extends Entity implements GeoEntity, OwnableEntity { protected static final EntityDataAccessor> OWNER_UUID = SynchedEntityData.defineId(Tm62Entity.class, EntityDataSerializers.OPTIONAL_UUID); protected static final EntityDataAccessor LAST_ATTACKER_UUID = SynchedEntityData.defineId(Tm62Entity.class, EntityDataSerializers.STRING); public static final EntityDataAccessor HEALTH = SynchedEntityData.defineId(Tm62Entity.class, EntityDataSerializers.FLOAT); public static final EntityDataAccessor FUSE = SynchedEntityData.defineId(Tm62Entity.class, EntityDataSerializers.BOOLEAN); private final AnimatableInstanceCache cache = GeckoLibUtil.createInstanceCache(this); public Tm62Entity(EntityType type, Level world) { super(type, world); } public Tm62Entity(LivingEntity owner, Level level, boolean fuse) { super(ModEntities.TM_62.get(), level); if (owner != null) { this.setOwnerUUID(owner.getUUID()); } this.entityData.set(FUSE, fuse); } @Override protected void defineSynchedData(SynchedEntityData.Builder builder) { builder.define(OWNER_UUID, Optional.empty()) .define(LAST_ATTACKER_UUID, "undefined") .define(FUSE, false) .define(HEALTH, 100f); } @Override public boolean isPickable() { return !this.isRemoved(); } private static final DamageModifier DAMAGE_MODIFIER = DamageModifier.createDefaultModifier() .multiply(0.02f, ModDamageTypes.CUSTOM_EXPLOSION) .multiply(0.02f, ModDamageTypes.MINE) .multiply(0.02f, ModDamageTypes.PROJECTILE_BOOM) .multiply(0.02f, DamageTypes.EXPLOSION); @Override public boolean hurt(@NotNull DamageSource source, float amount) { amount = DAMAGE_MODIFIER.compute(source, amount); if (source.getEntity() != null) { this.entityData.set(LAST_ATTACKER_UUID, source.getEntity().getStringUUID()); } this.entityData.set(HEALTH, this.entityData.get(HEALTH) - amount); return super.hurt(source, amount); } public void setOwnerUUID(@Nullable UUID pUuid) { this.entityData.set(OWNER_UUID, Optional.ofNullable(pUuid)); } @Nullable public UUID getOwnerUUID() { return this.entityData.get(OWNER_UUID).orElse(null); } public boolean isOwnedBy(LivingEntity pEntity) { return pEntity == this.getOwner(); } @Override public void addAdditionalSaveData(CompoundTag compound) { compound.putFloat("Health", this.entityData.get(HEALTH)); compound.putString("LastAttacker", this.entityData.get(LAST_ATTACKER_UUID)); compound.putBoolean("Fuse", this.entityData.get(FUSE)); if (this.getOwnerUUID() != null) { compound.putUUID("Owner", this.getOwnerUUID()); } } @Override public void readAdditionalSaveData(CompoundTag compound) { if (compound.contains("Health")) { this.entityData.set(HEALTH, compound.getFloat("Health")); } if (compound.contains("LastAttacker")) { this.entityData.set(LAST_ATTACKER_UUID, compound.getString("LastAttacker")); } if (compound.contains("Fuse")) { this.entityData.set(FUSE, compound.getBoolean("Fuse")); } UUID uuid; if (compound.hasUUID("Owner")) { uuid = compound.getUUID("Owner"); } else { String s = compound.getString("Owner"); assert this.getServer() != null; uuid = OldUsersConverter.convertMobOwnerIfNecessary(this.getServer(), s); } if (uuid != null) { try { this.setOwnerUUID(uuid); } catch (Throwable ignored) { } } } @Override @ParametersAreNonnullByDefault public @NotNull InteractionResult interact(Player player, InteractionHand hand) { if (this.isOwnedBy(player) && player.isShiftKeyDown()) { if (!this.level().isClientSide()) { this.discard(); } if (!player.getAbilities().instabuild) { ItemHandlerHelper.giveItemToPlayer(player, new ItemStack(ModItems.TM_62.get())); } } return InteractionResult.sidedSuccess(this.level().isClientSide()); } @Override public void tick() { super.tick(); if (this.tickCount >= 20 && onGround() && !entityData.get(FUSE)) { touchEntity(); } this.setDeltaMovement(this.getDeltaMovement().add(0.0, -0.03, 0.0)); if (!this.level().noCollision(this.getBoundingBox())) { this.moveTowardsClosestSpace(this.getX(), (this.getBoundingBox().minY + this.getBoundingBox().maxY) / 2.0, this.getZ()); } this.move(MoverType.SELF, this.getDeltaMovement()); float f = 0.98F; if (this.onGround()) { BlockPos pos = this.getBlockPosBelowThatAffectsMyMovement(); f = this.level().getBlockState(pos).getFriction(this.level(), pos, this) * 0.98F; } this.setDeltaMovement(this.getDeltaMovement().multiply(f, 0.98, f)); if (this.onGround()) { this.setDeltaMovement(this.getDeltaMovement().multiply(1.0, -0.9, 1.0)); } if (entityData.get(FUSE) && this.level() instanceof ServerLevel serverLevel) { ParticleTool.sendParticle(serverLevel, ParticleTypes.SMOKE, this.xo, this.yo, this.zo, 1, 0, 0, 0, 0.01, true); } if (this.entityData.get(HEALTH) <= 0 || (entityData.get(FUSE) && tickCount >= 100)) { triggerExplode(); } this.refreshDimensions(); } public void touchEntity() { if (level() instanceof ServerLevel) { var frontBox = getBoundingBox().inflate(0.2); boolean trigger = false; var entities = level().getEntities(EntityTypeTest.forClass(Entity.class), frontBox, entity -> entity != this && !(entity instanceof HangingEntity) && (entity.getBoundingBox().getSize() > 1.5 || (entity.getBoundingBox().getSize() > 0.9 && entity.getDeltaMovement().y() < -0.35))).stream().toList(); for (var entity : entities) { if (entity != null) { trigger = true; break; } } if (trigger) { triggerExplode(); if (this.level() instanceof ServerLevel) { AABB aabb = new AABB(position(), position()).inflate(2); BlockPos.betweenClosedStream(aabb).forEach((blockPos) -> { float hard = this.level().getBlockState(blockPos).getBlock().defaultDestroyTime(); if (ExplosionConfig.EXPLOSION_DESTROY.get() && hard != -1) { this.level().destroyBlock(blockPos, true); } }); } } } } private void triggerExplode() { CustomExplosion explosion = new CustomExplosion(this.level(), this, ModDamageTypes.causeCustomExplosionDamage(this.level().registryAccess(), this, this.getOwner()), 450f, this.getX(), this.getEyeY(), this.getZ(), 13f, ExplosionConfig.EXPLOSION_DESTROY.get() ? Explosion.BlockInteraction.DESTROY : Explosion.BlockInteraction.KEEP, true); explosion.explode(); EventHooks.onExplosionStart(this.level(), explosion); explosion.finalizeExplosion(false); ParticleTool.spawnHugeExplosionParticles(this.level(), this.position()); this.discard(); } @Override public boolean isPushable() { return super.isPushable(); } @Override public void registerControllers(AnimatableManager.ControllerRegistrar data) { } @Override public AnimatableInstanceCache getAnimatableInstanceCache() { return this.cache; } public void shoot(double pX, double pY, double pZ, float pVelocity, float pInaccuracy) { Vec3 vec3 = (new Vec3(pX, pY, pZ)).normalize().add(this.random.triangle(0.0, 0.0172275 * (double) pInaccuracy), this.random.triangle(0.0, 0.0172275 * (double) pInaccuracy), this.random.triangle(0.0, 0.0172275 * (double) pInaccuracy)).scale(pVelocity); this.setDeltaMovement(vec3); } }