package de.ellpeck.naturesaura.entities; import de.ellpeck.naturesaura.api.aura.chunk.IAuraChunk; import de.ellpeck.naturesaura.items.ModItems; import de.ellpeck.naturesaura.packet.PacketHandler; import de.ellpeck.naturesaura.packet.PacketParticles; import net.minecraft.core.BlockPos; import net.minecraft.nbt.CompoundTag; import net.minecraft.nbt.ListTag; import net.minecraft.nbt.LongTag; import net.minecraft.nbt.Tag; import net.minecraft.network.protocol.Packet; import net.minecraft.server.level.ServerLevel; import net.minecraft.util.Mth; import net.minecraft.world.damagesource.DamageSource; import net.minecraft.world.entity.Entity; import net.minecraft.world.entity.EntityType; import net.minecraft.world.entity.vehicle.AbstractMinecart; import net.minecraft.world.item.ItemStack; import net.minecraft.world.level.GameRules; import net.minecraft.world.level.Level; import net.minecraft.world.level.block.Blocks; import net.minecraft.world.level.block.state.BlockState; import net.minecraft.world.phys.HitResult; import net.minecraft.world.phys.Vec3; import net.minecraftforge.common.util.ITeleporter; import net.minecraftforge.network.NetworkHooks; import javax.annotation.Nullable; import java.util.ArrayList; import java.util.List; public class EntityMoverMinecart extends AbstractMinecart { private final List spotOffsets = new ArrayList<>(); public boolean isActive; private BlockPos lastPosition = BlockPos.ZERO; public EntityMoverMinecart(EntityType type, Level level) { super(type, level); } public EntityMoverMinecart(EntityType type, Level level, double x, double y, double z) { super(type, level, x, y, z); } @Override public void moveMinecartOnRail(BlockPos railPos) { super.moveMinecartOnRail(railPos); if (!this.isActive) return; BlockPos pos = this.blockPosition(); if (!this.spotOffsets.isEmpty() && this.level.getGameTime() % 10 == 0) PacketHandler.sendToAllAround(this.level, pos, 32, new PacketParticles( (float) this.getX(), (float) this.getY(), (float) this.getZ(), PacketParticles.Type.MOVER_CART, Mth.floor(this.getDeltaMovement().x * 100F), Mth.floor(this.getDeltaMovement().y * 100F), Mth.floor(this.getDeltaMovement().z * 100F))); if (pos.distSqr(this.lastPosition) < 8 * 8) return; this.moveAura(this.level, this.lastPosition, this.level, pos); this.lastPosition = pos; } private void moveAura(Level oldLevel, BlockPos oldPos, Level newLevel, BlockPos newPos) { for (BlockPos offset : this.spotOffsets) { BlockPos spot = oldPos.offset(offset); IAuraChunk chunk = IAuraChunk.getAuraChunk(oldLevel, spot); int amount = chunk.getDrainSpot(spot); if (amount <= 0) continue; int toMove = Math.min(amount, 300000); int drained = chunk.drainAura(spot, toMove, false, false); if (drained <= 0) continue; int toLose = Mth.ceil(drained / 250F); BlockPos newSpot = newPos.offset(offset); IAuraChunk.getAuraChunk(newLevel, newSpot).storeAura(newSpot, drained - toLose, false, false); } } @Override public void activateMinecart(int x, int y, int z, boolean receivingPower) { if (this.isActive != receivingPower) { this.isActive = receivingPower; BlockPos pos = this.blockPosition(); if (!this.isActive) { this.moveAura(this.level, this.lastPosition, this.level, pos); this.spotOffsets.clear(); this.lastPosition = BlockPos.ZERO; return; } IAuraChunk.getSpotsInArea(this.level, pos, 25, (spot, amount) -> { if (amount > 0) this.spotOffsets.add(spot.subtract(pos)); }); this.lastPosition = pos; } } @Override public void destroy(DamageSource source) { this.kill(); if (this.level.getGameRules().getBoolean(GameRules.RULE_DOENTITYDROPS)) this.spawnAtLocation(new ItemStack(ModItems.MOVER_CART), 0); } @Override public CompoundTag serializeNBT() { CompoundTag compound = super.serializeNBT(); compound.putBoolean("active", this.isActive); compound.putLong("last_pos", this.lastPosition.asLong()); ListTag list = new ListTag(); for (BlockPos offset : this.spotOffsets) list.add(LongTag.valueOf(offset.asLong())); compound.put("offsets", list); return compound; } @Override public void deserializeNBT(CompoundTag compound) { super.deserializeNBT(compound); this.isActive = compound.getBoolean("active"); this.lastPosition = BlockPos.of(compound.getLong("last_pos")); this.spotOffsets.clear(); ListTag list = compound.getList("offsets", Tag.TAG_LONG); for (Tag base : list) this.spotOffsets.add(BlockPos.of(((LongTag) base).getAsLong())); } @Nullable @Override public Entity changeDimension(ServerLevel destination, ITeleporter teleporter) { Entity entity = super.changeDimension(destination, teleporter); if (entity instanceof EntityMoverMinecart) { BlockPos pos = entity.blockPosition(); this.moveAura(this.level, this.lastPosition, entity.level, pos); ((EntityMoverMinecart) entity).lastPosition = pos; } return entity; } @Override public BlockState getDisplayBlockState() { return Blocks.STONE.defaultBlockState(); } @Override public Type getMinecartType() { return Type.RIDEABLE; } @Override public ItemStack getCartItem() { return new ItemStack(ModItems.MOVER_CART); } @Override public ItemStack getPickedResult(HitResult target) { return new ItemStack(ModItems.MOVER_CART); } @Override public boolean canBeRidden() { return false; } @Override protected void applyNaturalSlowdown() { Vec3 motion = this.getDeltaMovement(); this.setDeltaMovement(motion.x * 0.99F, 0, motion.z * 0.99F); } @Override public Packet getAddEntityPacket() { return NetworkHooks.getEntitySpawningPacket(this); } }