package com.atsuishio.superbwarfare.entity.vehicle; import com.atsuishio.superbwarfare.Mod; import com.atsuishio.superbwarfare.config.server.ExplosionConfig; import com.atsuishio.superbwarfare.config.server.VehicleConfig; import com.atsuishio.superbwarfare.entity.projectile.*; import com.atsuishio.superbwarfare.entity.vehicle.base.ContainerMobileVehicleEntity; import com.atsuishio.superbwarfare.entity.vehicle.base.HelicopterEntity; import com.atsuishio.superbwarfare.entity.vehicle.base.ThirdPersonCameraPosition; import com.atsuishio.superbwarfare.entity.vehicle.base.WeaponVehicleEntity; import com.atsuishio.superbwarfare.entity.vehicle.damage.DamageModifier; import com.atsuishio.superbwarfare.entity.vehicle.weapon.HeliRocketWeapon; import com.atsuishio.superbwarfare.entity.vehicle.weapon.ProjectileWeapon; import com.atsuishio.superbwarfare.entity.vehicle.weapon.VehicleWeapon; import com.atsuishio.superbwarfare.init.ModDamageTypes; import com.atsuishio.superbwarfare.init.ModItems; import com.atsuishio.superbwarfare.init.ModSounds; import com.atsuishio.superbwarfare.init.ModTags; import com.atsuishio.superbwarfare.network.message.receive.ShakeClientMessage; import com.atsuishio.superbwarfare.tools.AmmoType; import com.atsuishio.superbwarfare.tools.CustomExplosion; import com.atsuishio.superbwarfare.tools.InventoryTool; import com.atsuishio.superbwarfare.tools.ParticleTool; import com.google.common.collect.Lists; import com.mojang.math.Axis; import net.minecraft.core.BlockPos; import net.minecraft.core.particles.ParticleTypes; import net.minecraft.nbt.CompoundTag; import net.minecraft.network.protocol.Packet; import net.minecraft.network.protocol.game.ClientGamePacketListener; import net.minecraft.network.syncher.EntityDataAccessor; import net.minecraft.network.syncher.EntityDataSerializers; import net.minecraft.network.syncher.SynchedEntityData; import net.minecraft.resources.ResourceLocation; import net.minecraft.server.level.ServerEntity; import net.minecraft.server.level.ServerLevel; import net.minecraft.server.level.ServerPlayer; import net.minecraft.sounds.SoundEvent; import net.minecraft.util.Mth; import net.minecraft.world.damagesource.DamageTypes; import net.minecraft.world.entity.Entity; import net.minecraft.world.entity.EntityType; import net.minecraft.world.entity.LivingEntity; import net.minecraft.world.entity.Pose; import net.minecraft.world.entity.player.Player; import net.minecraft.world.entity.vehicle.DismountHelper; import net.minecraft.world.item.ItemStack; import net.minecraft.world.level.Explosion; import net.minecraft.world.level.Level; import net.minecraft.world.phys.AABB; import net.minecraft.world.phys.Vec3; import net.neoforged.neoforge.event.EventHooks; import net.neoforged.neoforge.network.PacketDistributor; import org.jetbrains.annotations.NotNull; import org.joml.Math; import org.joml.Matrix4f; import org.joml.Vector3f; import org.joml.Vector4f; 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 java.util.ArrayList; import java.util.Comparator; import static com.atsuishio.superbwarfare.tools.ParticleTool.sendParticle; public class Ah6Entity extends ContainerMobileVehicleEntity implements GeoEntity, HelicopterEntity, WeaponVehicleEntity { private final AnimatableInstanceCache cache = GeckoLibUtil.createInstanceCache(this); public static final EntityDataAccessor PROPELLER_ROT = SynchedEntityData.defineId(Ah6Entity.class, EntityDataSerializers.FLOAT); public static final EntityDataAccessor DECOY_COUNT = SynchedEntityData.defineId(Ah6Entity.class, EntityDataSerializers.INT); public static final EntityDataAccessor LOADED_ROCKET = SynchedEntityData.defineId(Ah6Entity.class, EntityDataSerializers.INT); public boolean engineStart; public boolean engineStartOver; public double velocity; public int decoyReloadCoolDown; public int fireIndex; public int holdTick; public int holdPowerTick; public Ah6Entity(EntityType pEntityType, Level pLevel) { super(pEntityType, pLevel); } @Override public float maxUpStep() { return 1.1F; } @Override public VehicleWeapon[][] initWeapons() { return new VehicleWeapon[][]{ new VehicleWeapon[]{ new ProjectileWeapon() .damage(VehicleConfig.AH_6_CANNON_DAMAGE.get()) .headShot(2) .zoom(false) .heBullet(1) .bypassArmorRate(0.1f) .sound(ModSounds.INTO_CANNON.get()) .icon(Mod.loc("textures/screens/vehicle_weapon/cannon_20mm.png")), new HeliRocketWeapon() .damage(VehicleConfig.AH_6_ROCKET_DAMAGE.get()) .explosionDamage(VehicleConfig.AH_6_ROCKET_EXPLOSION_DAMAGE.get()) .explosionRadius(VehicleConfig.AH_6_ROCKET_EXPLOSION_RADIUS.get()) .sound(ModSounds.INTO_MISSILE.get()), } }; } @Override public ThirdPersonCameraPosition getThirdPersonCameraPosition(int index) { return new ThirdPersonCameraPosition(7, 1, -2.7); } @Override protected void defineSynchedData(SynchedEntityData.Builder builder) { super.defineSynchedData(builder); builder.define(LOADED_ROCKET, 0) .define(PROPELLER_ROT, 0f) .define(DECOY_COUNT, 6); } @Override public void addAdditionalSaveData(CompoundTag compound) { super.addAdditionalSaveData(compound); compound.putInt("LoadedRocket", this.entityData.get(LOADED_ROCKET)); compound.putFloat("PropellerRot", this.entityData.get(PROPELLER_ROT)); compound.putInt("DecoyCount", this.entityData.get(DECOY_COUNT)); } @Override public void readAdditionalSaveData(CompoundTag compound) { super.readAdditionalSaveData(compound); this.entityData.set(LOADED_ROCKET, compound.getInt("LoadedRocket")); this.entityData.set(PROPELLER_ROT, compound.getFloat("PropellerRot")); this.entityData.set(DECOY_COUNT, compound.getInt("DecoyCount")); } // TODO addentitypacket @Override public @NotNull Packet getAddEntityPacket(@NotNull ServerEntity entity) { return super.getAddEntityPacket(entity); } // @Override // public Packet getAddEntityPacket() { // return NetworkHooks.getEntitySpawningPacket(this); // } @Override public DamageModifier getDamageModifier() { return super.getDamageModifier() .multiply(0.5f) .multiply(0.2f, DamageTypes.ARROW) .multiply(0.4f, DamageTypes.TRIDENT) .multiply(0.4f, DamageTypes.MOB_ATTACK) .multiply(0.4f, DamageTypes.MOB_ATTACK_NO_AGGRO) .multiply(0.4f, DamageTypes.MOB_PROJECTILE) .multiply(0.4f, DamageTypes.PLAYER_ATTACK) .multiply(4, DamageTypes.LAVA) .multiply(4, DamageTypes.EXPLOSION) .multiply(4, DamageTypes.PLAYER_EXPLOSION) .multiply(0.8f, ModDamageTypes.CANNON_FIRE) .multiply(0.16f, ModTags.DamageTypes.PROJECTILE) .multiply(10, ModDamageTypes.VEHICLE_STRIKE) .custom((source, damage) -> { if (source.getDirectEntity() instanceof MelonBombEntity) { return 2f * damage; } if (source.getDirectEntity() instanceof RgoGrenadeEntity) { return 6f * damage; } if (source.getDirectEntity() instanceof HandGrenadeEntity) { return 5f * damage; } if (source.getDirectEntity() instanceof MortarShellEntity) { return 4f * damage; } return damage; }) .reduce(2); } @Override public void baseTick() { super.baseTick(); if (this.level() instanceof ServerLevel) { if (reloadCoolDown > 0) { reloadCoolDown--; } if (decoyReloadCoolDown > 0) { decoyReloadCoolDown--; } handleAmmo(); } if (this.onGround()) { this.terrainCompat(2.7f, 2.7f); this.setDeltaMovement(this.getDeltaMovement().multiply(0.8, 1, 0.8)); } else { setZRot(getRoll() * (backInputDown ? 0.9f : 0.99f)); float f = (float) Mth.clamp(0.9f - 0.015 * getDeltaMovement().length() + 0.02f * Mth.abs(90 - (float) calculateAngle(this.getDeltaMovement(), this.getViewVector(1))) / 90, 0.01, 0.99); this.setDeltaMovement(this.getDeltaMovement().add(this.getViewVector(1).scale((this.getXRot() < 0 ? -0.035 : (this.getXRot() > 0 ? 0.035 : 0)) * this.getDeltaMovement().length()))); this.setDeltaMovement(this.getDeltaMovement().multiply(f, 0.95, f)); } if (this.isInWater() && this.tickCount % 4 == 0 && getSubmergedHeight(this) > 0.5 * getBbHeight()) { this.setDeltaMovement(this.getDeltaMovement().multiply(0.6, 0.6, 0.6)); this.hurt(ModDamageTypes.causeVehicleStrikeDamage(this.level().registryAccess(), this, this.getFirstPassenger() == null ? this : this.getFirstPassenger()), 1 + (float) (20 * ((lastTickSpeed - 0.4) * (lastTickSpeed - 0.4)))); } releaseDecoy(); lowHealthWarning(); this.refreshDimensions(); } private void handleAmmo() { if (!(this.getFirstPassenger() instanceof Player player)) return; int ammoCount = this.getItemStacks().stream().filter(stack -> { if (stack.is(ModItems.AMMO_BOX.get())) { return AmmoType.HEAVY.get(stack) > 0; } return false; }).mapToInt(AmmoType.HEAVY::get).sum() + countItem(ModItems.HEAVY_AMMO.get()); if ((hasItem(ModItems.ROCKET_70.get()) || InventoryTool.hasCreativeAmmoBox(player)) && reloadCoolDown == 0 && this.getEntityData().get(LOADED_ROCKET) < 14) { this.entityData.set(LOADED_ROCKET, this.getEntityData().get(LOADED_ROCKET) + 1); reloadCoolDown = 25; if (!InventoryTool.hasCreativeAmmoBox(player)) { this.getItemStacks().stream().filter(stack -> stack.is(ModItems.ROCKET_70.get())).findFirst().ifPresent(stack -> stack.shrink(1)); } this.level().playSound(null, this, ModSounds.MISSILE_RELOAD.get(), this.getSoundSource(), 1, 1); } if (this.getWeaponIndex(0) == 0) { this.entityData.set(AMMO, ammoCount); } else { this.entityData.set(AMMO, this.getEntityData().get(LOADED_ROCKET)); } } public void releaseDecoy() { if (decoyInputDown) { if (this.entityData.get(DECOY_COUNT) > 0 && this.level() instanceof ServerLevel) { Entity passenger = getFirstPassenger(); for (int i = 0; i < 4; i++) { FlareDecoyEntity flareDecoyEntity = new FlareDecoyEntity((LivingEntity) passenger, this.level()); flareDecoyEntity.setPos(this.getX() + this.getDeltaMovement().x, this.getY() + 0.5 + this.getDeltaMovement().y, this.getZ() + this.getDeltaMovement().z); flareDecoyEntity.decoyShoot(this, this.getViewVector(1).yRot((45 + 90 * i) * Mth.DEG_TO_RAD), 0.8f, 8); this.level().addFreshEntity(flareDecoyEntity); } this.level().playSound(null, this, ModSounds.DECOY_FIRE.get(), this.getSoundSource(), 1, 1); if (this.getEntityData().get(DECOY_COUNT) == 6) { decoyReloadCoolDown = 300; } this.getEntityData().set(DECOY_COUNT, this.getEntityData().get(DECOY_COUNT) - 1); } decoyInputDown = false; } if (this.entityData.get(DECOY_COUNT) < 6 && decoyReloadCoolDown == 0 && this.level() instanceof ServerLevel) { this.entityData.set(DECOY_COUNT, this.entityData.get(DECOY_COUNT) + 1); this.level().playSound(null, this, ModSounds.DECOY_RELOAD.get(), this.getSoundSource(), 1, 1); decoyReloadCoolDown = 300; } } @Override public void travel() { Entity passenger = getFirstPassenger(); Entity passenger2 = getNthEntity(1); Entity passenger3 = getNthEntity(2); Entity passenger4 = getNthEntity(3); float diffX; float diffY; if (passenger == null) { this.leftInputDown = false; this.rightInputDown = false; this.forwardInputDown = false; this.backInputDown = false; this.upInputDown = false; this.downInputDown = false; this.setZRot(this.roll * 0.98f); this.setXRot(this.getXRot() * 0.98f); if (passenger2 == null && passenger3 == null && passenger4 == null) { this.entityData.set(POWER, this.entityData.get(POWER) * 0.99f); } } else if (passenger instanceof Player) { diffY = Math.clamp(-90f, 90f, Mth.wrapDegrees(passenger.getYHeadRot() - this.getYRot())); diffX = Math.clamp(-60f, 60f, Mth.wrapDegrees(passenger.getXRot() - this.getXRot())); if (rightInputDown) { holdTick++; this.entityData.set(DELTA_ROT, this.entityData.get(DELTA_ROT) - 2f * Math.min(holdTick, 7) * this.entityData.get(POWER)); } else if (this.leftInputDown) { holdTick++; this.entityData.set(DELTA_ROT, this.entityData.get(DELTA_ROT) + 2f * Math.min(holdTick, 7) * this.entityData.get(POWER)); } else { holdTick = 0; } this.setYRot(this.getYRot() + Mth.clamp((this.onGround() ? 0.1f : 2f) * diffY * this.entityData.get(PROPELLER_ROT), -10f, 10f)); this.setXRot(Mth.clamp(this.getXRot() + ((this.onGround()) ? 0 : 1.5f) * diffX * this.entityData.get(PROPELLER_ROT), -80, 80)); this.setZRot(this.getRoll() - this.entityData.get(DELTA_ROT) + (this.onGround() ? 0 : 0.25f) * diffY * this.entityData.get(PROPELLER_ROT)); } if (this.level() instanceof ServerLevel) { if (this.getEnergy() > 0) { boolean up = upInputDown || forwardInputDown; boolean down = this.downInputDown; if (!engineStart && up) { engineStart = true; this.level().playSound(null, this, ModSounds.HELICOPTER_ENGINE_START.get(), this.getSoundSource(), 3, 1); } if (up && engineStartOver) { holdPowerTick++; this.entityData.set(POWER, Math.min(this.entityData.get(POWER) + 0.0007f * Math.min(holdPowerTick, 10), 0.12f)); } if (engineStartOver) { if (down) { holdPowerTick++; this.entityData.set(POWER, Math.max(this.entityData.get(POWER) - 0.001f * Math.min(holdPowerTick, 5), this.onGround() ? 0 : 0.025f)); } else if (backInputDown) { holdPowerTick++; this.entityData.set(POWER, Math.max(this.entityData.get(POWER) - 0.001f * Math.min(holdPowerTick, 5), this.onGround() ? 0 : 0.052f)); if (passenger != null) { passenger.setXRot(0.8f * passenger.getXRot()); } } } if (engineStart && !engineStartOver) { this.entityData.set(POWER, Math.min(this.entityData.get(POWER) + 0.0012f, 0.045f)); } if (!(up || down || backInputDown) && engineStartOver) { if (this.getDeltaMovement().y() < 0) { this.entityData.set(POWER, Math.min(this.entityData.get(POWER) + 0.0002f, 0.12f)); } else { this.entityData.set(POWER, Math.max(this.entityData.get(POWER) - (this.onGround() ? 0.00005f : 0.0002f), 0)); } holdPowerTick = 0; } } else { this.entityData.set(POWER, Math.max(this.entityData.get(POWER) - 0.0001f, 0)); this.forwardInputDown = false; this.backInputDown = false; engineStart = false; engineStartOver = false; } } this.entityData.set(DELTA_ROT, this.entityData.get(DELTA_ROT) * 0.9f); this.entityData.set(PROPELLER_ROT, Mth.lerp(0.18f, this.entityData.get(PROPELLER_ROT), this.entityData.get(POWER))); this.setPropellerRot(this.getPropellerRot() + 30 * this.entityData.get(PROPELLER_ROT)); this.entityData.set(PROPELLER_ROT, this.entityData.get(PROPELLER_ROT) * 0.9995f); if (engineStart) { this.consumeEnergy((int) (VehicleConfig.AH_6_MIN_ENERGY_COST.get() + this.entityData.get(POWER) * ((VehicleConfig.AH_6_MAX_ENERGY_COST.get() - VehicleConfig.AH_6_MIN_ENERGY_COST.get()) / 0.12))); } setDeltaMovement(getDeltaMovement().add(0.0f, Math.min(Math.sin((90 - this.getXRot()) * Mth.DEG_TO_RAD), Math.sin((90 + this.getRoll()) * Mth.DEG_TO_RAD)) * this.entityData.get(POWER), 0.0f)); Vector3f direction = getRightDirection().mul(-Math.sin(this.getRoll() * Mth.DEG_TO_RAD) * this.entityData.get(PROPELLER_ROT)); setDeltaMovement(getDeltaMovement().add(new Vec3(direction.x, direction.y, direction.z).scale(3))); // if (passenger instanceof Player player) { // player.displayClientMessage(Component.literal(this.getRoll() + ""), true); // } Vector3f directionZ = getForwardDirection().mul(-Math.cos((this.getXRot() + 90) * Mth.DEG_TO_RAD) * this.entityData.get(PROPELLER_ROT)); setDeltaMovement(getDeltaMovement().add(new Vec3(directionZ.x, directionZ.y, directionZ.z).scale(3))); if (this.entityData.get(POWER) > 0.04f) { engineStartOver = true; } if (this.entityData.get(POWER) < 0.0004f) { engineStart = false; engineStartOver = false; } } @Override public SoundEvent getEngineSound() { return ModSounds.HELICOPTER_ENGINE.get(); } protected void clampRotation(Entity entity) { if (entity == getNthEntity(0) || entity == getNthEntity(1)) { float f = Mth.wrapDegrees(entity.getXRot()); float f1 = Mth.clamp(f, -80.0F, 80F); entity.xRotO += f1 - f; entity.setXRot(entity.getXRot() + f1 - f); float f2 = Mth.wrapDegrees(entity.getYRot() - this.getYRot()); float f3 = Mth.clamp(f2, -80.0F, 80.0F); entity.yRotO += f3 - f2; entity.setYRot(entity.getYRot() + f3 - f2); entity.setYBodyRot(this.getYRot()); } else if (entity == getNthEntity(2)) { float f2 = Mth.wrapDegrees(entity.getYRot() - this.getYRot()); float f3 = Mth.clamp(f2, 10.0F, 170.0F); entity.yRotO += f3 - f2; entity.setYRot(entity.getYRot() + f3 - f2); entity.setYBodyRot(getYRot() + 90); } else if (entity == getNthEntity(3)) { float f2 = Mth.wrapDegrees(entity.getYRot() - this.getYRot()); float f3 = Mth.clamp(f2, -170.0F, -10.0F); entity.yRotO += f3 - f2; entity.setYRot(entity.getYRot() + f3 - f2); entity.setYBodyRot(getYRot() - 90); } } @Override public void onPassengerTurned(@NotNull Entity entity) { this.clampRotation(entity); } @Override public void positionRider(@NotNull Entity passenger, @NotNull MoveFunction callback) { // From Immersive_Aircraft if (!this.hasPassenger(passenger)) { return; } Matrix4f transform = getVehicleTransform(1); float x = 0.6f; float y = -0.65f; float z = 1f; int i = this.getOrderedPassengers().indexOf(passenger); if (i == 0) { Vector4f worldPosition = transformPosition(transform, x, y, z); passenger.setPos(worldPosition.x, worldPosition.y, worldPosition.z); callback.accept(passenger, worldPosition.x, worldPosition.y, worldPosition.z); } else if (i == 1) { Vector4f worldPosition = transformPosition(transform, -x, y, z); passenger.setPos(worldPosition.x, worldPosition.y, worldPosition.z); callback.accept(passenger, worldPosition.x, worldPosition.y, worldPosition.z); } else if (i == 2) { Vector4f worldPosition = transformPosition(transform, -1.4f, -1.05f, 0); passenger.setPos(worldPosition.x, worldPosition.y, worldPosition.z); callback.accept(passenger, worldPosition.x, worldPosition.y, worldPosition.z); } else if (i == 3) { Vector4f worldPosition = transformPosition(transform, 1.4f, -1.05f, 0); passenger.setPos(worldPosition.x, worldPosition.y, worldPosition.z); callback.accept(passenger, worldPosition.x, worldPosition.y, worldPosition.z); } if (passenger != this.getFirstPassenger()) { passenger.setXRot(passenger.getXRot() + (getXRot() - xRotO)); } copyEntityData(passenger); } public void copyEntityData(Entity entity) { if (entity == getNthEntity(0) || entity == getNthEntity(1)) { float f = Mth.wrapDegrees(entity.getYRot() - getYRot()); float g = Mth.clamp(f, -105.0f, 105.0f); entity.yRotO += g - f; entity.setYRot(entity.getYRot() + g - f); entity.setYHeadRot(entity.getYRot()); entity.setYBodyRot(getYRot()); } else if (entity == getNthEntity(2)) { float f = Mth.wrapDegrees(entity.getYRot() - getYRot()); float g = Mth.clamp(f, 10.0f, 170.0f); entity.yRotO += g - f; entity.setYRot(entity.getYRot() + g - f); entity.setYHeadRot(entity.getYRot()); entity.setYBodyRot(getYRot() + 90); } else if (entity == getNthEntity(3)) { float f = Mth.wrapDegrees(entity.getYRot() - getYRot()); float g = Mth.clamp(f, -170.0f, -10.0f); entity.yRotO += g - f; entity.setYRot(entity.getYRot() + g - f); entity.setYHeadRot(entity.getYRot()); entity.setYBodyRot(getYRot() - 90); } } @Override public Matrix4f getVehicleTransform(float ticks) { Matrix4f transform = new Matrix4f(); transform.translate((float) Mth.lerp(ticks, xo, getX()), (float) Mth.lerp(ticks, yo + 1.45f, getY() + 1.45f), (float) Mth.lerp(ticks, zo, getZ())); transform.rotate(Axis.YP.rotationDegrees(-Mth.lerp(ticks, yRotO, getYRot()))); transform.rotate(Axis.XP.rotationDegrees(Mth.lerp(ticks, xRotO, getXRot()))); transform.rotate(Axis.ZP.rotationDegrees(Mth.lerp(ticks, prevRoll, getRoll()))); return transform; } @Override public void destroy() { if (this.crash) { crashPassengers(); } else { explodePassengers(); } if (level() instanceof ServerLevel) { CustomExplosion explosion = new CustomExplosion(this.level(), this, ModDamageTypes.causeCustomExplosionDamage(this.level().registryAccess(), this, getAttacker()), 300.0f, this.getX(), this.getY(), this.getZ(), 8f, ExplosionConfig.EXPLOSION_DESTROY.get() ? Explosion.BlockInteraction.DESTROY : Explosion.BlockInteraction.KEEP).setDamageMultiplier(1); explosion.explode(); EventHooks.onExplosionStart(this.level(), explosion); explosion.finalizeExplosion(false); ParticleTool.spawnHugeExplosionParticles(this.level(), this.position()); } this.discard(); } @Override public void registerControllers(AnimatableManager.ControllerRegistrar data) { } @Override public AnimatableInstanceCache getAnimatableInstanceCache() { return this.cache; } @Override public float getMaxHealth() { return VehicleConfig.AH_6_HP.get(); } @Override public int getMaxEnergy() { return VehicleConfig.AH_6_MAX_ENERGY.get(); } @Override public void vehicleShoot(Player player, int type) { boolean hasCreativeAmmo = false; for (int i = 0; i < getMaxPassengers() - 1; i++) { if (getNthEntity(i) instanceof Player pPlayer && InventoryTool.hasCreativeAmmoBox(pPlayer)) { hasCreativeAmmo = true; } } Matrix4f transform = getVehicleTransform(1); float x; float y; float z; Vector4f worldPositionRight; Vector4f worldPositionLeft; if (getWeaponIndex(0) == 0) { if (this.cannotFire) return; x = 1.15f; y = 0.62f - 1.45f; z = 0.8f; worldPositionRight = transformPosition(transform, -x, y, z); worldPositionLeft = transformPosition(transform, x, y, z); if (this.entityData.get(AMMO) > 0 || hasCreativeAmmo) { ProjectileEntity projectileRight = ((ProjectileWeapon) getWeapon(0)).create(player); projectileRight.setPos(worldPositionRight.x, worldPositionRight.y, worldPositionRight.z); projectileRight.shoot(player, this.getLookAngle().x, this.getLookAngle().y + 0.018, this.getLookAngle().z, 20, (float) 0.2); this.level().addFreshEntity(projectileRight); sendParticle((ServerLevel) this.level(), ParticleTypes.LARGE_SMOKE, worldPositionRight.x, worldPositionRight.y, worldPositionRight.z, 1, 0, 0, 0, 0, false); if (!InventoryTool.hasCreativeAmmoBox(player)) { this.getItemStacks().stream().filter(stack -> stack.is(ModItems.HEAVY_AMMO.get())).findFirst().ifPresent(stack -> stack.shrink(1)); } } if (this.entityData.get(AMMO) > 0 || hasCreativeAmmo) { ProjectileEntity projectileLeft = ((ProjectileWeapon) getWeapon(0)).create(player); projectileLeft.setPos(worldPositionLeft.x, worldPositionLeft.y, worldPositionLeft.z); projectileLeft.shoot(player, this.getLookAngle().x, this.getLookAngle().y + 0.018, this.getLookAngle().z, 20, (float) 0.2); this.level().addFreshEntity(projectileLeft); sendParticle((ServerLevel) this.level(), ParticleTypes.LARGE_SMOKE, worldPositionLeft.x, worldPositionLeft.y, worldPositionLeft.z, 1, 0, 0, 0, 0, false); if (!hasCreativeAmmo) { ItemStack ammoBox = this.getItemStacks().stream().filter(stack -> { if (stack.is(ModItems.AMMO_BOX.get())) { return AmmoType.HEAVY.get(stack) > 0; } return false; }).findFirst().orElse(ItemStack.EMPTY); if (!ammoBox.isEmpty()) { AmmoType.HEAVY.add(ammoBox, -1); } else { this.getItemStacks().stream().filter(stack -> stack.is(ModItems.HEAVY_AMMO.get())).findFirst().ifPresent(stack -> stack.shrink(1)); } } } this.entityData.set(HEAT, this.entityData.get(HEAT) + 5); if (!player.level().isClientSide) { if (player instanceof ServerPlayer serverPlayer) { serverPlayer.playSound(ModSounds.HELICOPTER_CANNON_FIRE_3P.get(), 4, 1); serverPlayer.playSound(ModSounds.HELICOPTER_CANNON_FAR.get(), 12, 1); serverPlayer.playSound(ModSounds.HELICOPTER_CANNON_VERYFAR.get(), 24, 1); } } Level level = player.level(); final Vec3 center = new Vec3(this.getX(), this.getEyeY(), this.getZ()); for (Entity target : level.getEntitiesOfClass(Entity.class, new AABB(center, center).inflate(6), e -> true).stream().sorted(Comparator.comparingDouble(e -> e.distanceToSqr(center))).toList()) { if (target instanceof ServerPlayer serverPlayer) { PacketDistributor.sendToPlayer(serverPlayer, new ShakeClientMessage(6, 4, 8, this.getX(), this.getEyeY(), this.getZ())); } } } else if (getWeaponIndex(0) == 1 && this.getEntityData().get(LOADED_ROCKET) > 0) { x = 1.7f; y = 0.62f - 1.45f; z = 0.8f; worldPositionRight = transformPosition(transform, -x, y, z); worldPositionLeft = transformPosition(transform, x, y, z); var heliRocketEntity = ((HeliRocketWeapon) getWeapon(0)).create(player); if (fireIndex == 0) { heliRocketEntity.setPos(worldPositionRight.x, worldPositionRight.y, worldPositionRight.z); heliRocketEntity.shoot(this.getLookAngle().x, this.getLookAngle().y + 0.008, this.getLookAngle().z, 7, 0.25f); player.level().addFreshEntity(heliRocketEntity); fireIndex = 1; } else if (fireIndex == 1) { heliRocketEntity.setPos(worldPositionLeft.x, worldPositionLeft.y, worldPositionLeft.z); heliRocketEntity.shoot(this.getLookAngle().x, this.getLookAngle().y + 0.008, this.getLookAngle().z, 7, 0.25f); player.level().addFreshEntity(heliRocketEntity); fireIndex = 0; } if (!player.level().isClientSide) { if (player instanceof ServerPlayer serverPlayer) { serverPlayer.playSound(ModSounds.HELICOPTER_ROCKET_FIRE_3P.get(), 6, 1); } } this.entityData.set(LOADED_ROCKET, this.getEntityData().get(LOADED_ROCKET) - 1); reloadCoolDown = 30; Level level = player.level(); final Vec3 center = new Vec3(this.getX(), this.getEyeY(), this.getZ()); for (Entity target : level.getEntitiesOfClass(Entity.class, new AABB(center, center).inflate(6), e -> true).stream().sorted(Comparator.comparingDouble(e -> e.distanceToSqr(center))).toList()) { if (target instanceof ServerPlayer serverPlayer) { PacketDistributor.sendToPlayer(serverPlayer, new ShakeClientMessage(6, 5, 7, this.getX(), this.getEyeY(), this.getZ())); } } } } @Override public @NotNull Vec3 getDismountLocationForPassenger(LivingEntity passenger) { Vec3 vec3d = getDismountOffset(getBbWidth() * Mth.SQRT_OF_TWO, passenger.getBbWidth() * Mth.SQRT_OF_TWO); double ox = getX() + vec3d.x; int i = this.getOrderedPassengers().indexOf(passenger); if (i == 0 || i == 2 || i == 3 || i == 4) { ox = getX() - vec3d.x; } double oz = getZ() + vec3d.z; BlockPos exitPos = new BlockPos((int) ox, (int) getY(), (int) oz); BlockPos floorPos = exitPos.below(); if (!level().isWaterAt(floorPos)) { ArrayList list = Lists.newArrayList(); double exitHeight = level().getBlockFloorHeight(exitPos); if (DismountHelper.isBlockFloorValid(exitHeight)) { list.add(new Vec3(ox, (double) exitPos.getY() + exitHeight, oz)); } double floorHeight = level().getBlockFloorHeight(floorPos); if (DismountHelper.isBlockFloorValid(floorHeight)) { list.add(new Vec3(ox, (double) floorPos.getY() + floorHeight, oz)); } for (Pose entityPose : passenger.getDismountPoses()) { for (Vec3 vec3d2 : list) { if (!DismountHelper.canDismountTo(level(), vec3d2, passenger, entityPose)) continue; passenger.setPose(entityPose); return vec3d2; } } } return super.getDismountLocationForPassenger(passenger); } @Override public int mainGunRpm(Player player) { return 360; } @Override public boolean canShoot(Player player) { if (getWeaponIndex(0) == 0) { return (this.entityData.get(AMMO) > 0 || InventoryTool.hasCreativeAmmoBox(player)) && !cannotFire; } else if (getWeaponIndex(0) == 1) { return this.entityData.get(AMMO) > 0; } return false; } @Override public int getAmmoCount(Player player) { return this.entityData.get(AMMO); } @Override public boolean hidePassenger(Entity entity) { return false; } @Override public int zoomFov() { return 3; } @Override public float getRotX(float tickDelta) { return this.getPitch(tickDelta); } @Override public float getRotY(float tickDelta) { return this.getYaw(tickDelta); } @Override public float getRotZ(float tickDelta) { return this.getRoll(tickDelta); } @Override public float getPower() { return this.entityData.get(POWER); } @Override public int getDecoy() { return this.entityData.get(DECOY_COUNT); } public int getMaxPassengers() { return 4; } @Override public ResourceLocation getVehicleIcon() { return Mod.loc("textures/vehicle_icon/ah_6_icon.png"); } @Override public boolean allowFreeCam() { return true; } }