261 lines
No EOL
10 KiB
Java
261 lines
No EOL
10 KiB
Java
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<Optional<UUID>> OWNER_UUID = SynchedEntityData.defineId(Tm62Entity.class, EntityDataSerializers.OPTIONAL_UUID);
|
|
protected static final EntityDataAccessor<String> LAST_ATTACKER_UUID = SynchedEntityData.defineId(Tm62Entity.class, EntityDataSerializers.STRING);
|
|
public static final EntityDataAccessor<Float> HEALTH = SynchedEntityData.defineId(Tm62Entity.class, EntityDataSerializers.FLOAT);
|
|
|
|
public static final EntityDataAccessor<Boolean> FUSE = SynchedEntityData.defineId(Tm62Entity.class, EntityDataSerializers.BOOLEAN);
|
|
private final AnimatableInstanceCache cache = GeckoLibUtil.createInstanceCache(this);
|
|
|
|
public Tm62Entity(EntityType<Tm62Entity> 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);
|
|
}
|
|
} |