mirror of
https://github.com/HbmMods/Hbm-s-Nuclear-Tech-GIT.git
synced 2026-01-25 10:32:49 +00:00
287 lines
9.8 KiB
Java
287 lines
9.8 KiB
Java
package com.hbm.entity.effect;
|
|
|
|
import com.hbm.blocks.ModBlocks;
|
|
import com.hbm.config.BombConfig;
|
|
import com.hbm.config.RadiationConfig;
|
|
import com.hbm.config.VersatileConfig;
|
|
import com.hbm.saveddata.AuxSavedData;
|
|
import net.minecraft.block.Block;
|
|
import net.minecraft.block.material.Material;
|
|
import net.minecraft.entity.Entity;
|
|
import net.minecraft.init.Blocks;
|
|
import net.minecraft.nbt.NBTTagCompound;
|
|
import net.minecraft.util.Vec3;
|
|
import net.minecraft.world.ChunkCoordIntPair;
|
|
import net.minecraft.world.World;
|
|
import net.minecraftforge.common.util.ForgeDirection;
|
|
|
|
import java.util.*;
|
|
|
|
public class EntityFalloutRain extends Entity {
|
|
private boolean firstTick = true; // Of course Vanilla has it private in Entity...
|
|
|
|
public EntityFalloutRain(World p_i1582_1_) {
|
|
super(p_i1582_1_);
|
|
this.setSize(4, 20);
|
|
this.ignoreFrustumCheck = true;
|
|
this.isImmuneToFire = true;
|
|
}
|
|
|
|
public EntityFalloutRain(World p_i1582_1_, int maxAge) {
|
|
super(p_i1582_1_);
|
|
this.setSize(4, 20);
|
|
this.isImmuneToFire = true;
|
|
}
|
|
|
|
private int tickDelay = BombConfig.fDelay;
|
|
|
|
@Override
|
|
public void onUpdate() {
|
|
if(!worldObj.isRemote) {
|
|
if (firstTick) {
|
|
if (chunksToProcess.isEmpty() && outerChunksToProcess.isEmpty()) gatherChunks();
|
|
firstTick = false;
|
|
}
|
|
|
|
if (tickDelay == 0) {
|
|
tickDelay = BombConfig.fDelay;
|
|
if (!chunksToProcess.isEmpty()) {
|
|
long chunkPos = chunksToProcess.remove(chunksToProcess.size() - 1); // Just so it doesn't shift the whole list every time
|
|
int chunkPosX = (int) (chunkPos & Integer.MAX_VALUE);
|
|
int chunkPosZ = (int) (chunkPos >> 32 & Integer.MAX_VALUE);
|
|
for (int x = chunkPosX << 4; x <= (chunkPosX << 4) + 16; x++) for (int z = chunkPosZ << 4; z <= (chunkPosZ << 4) + 16; z++)
|
|
stomp(x, z, Math.hypot(x - posX, z - posZ) * 100 / getScale());
|
|
} else if (!outerChunksToProcess.isEmpty()) {
|
|
long chunkPos = outerChunksToProcess.remove(outerChunksToProcess.size() - 1);
|
|
int chunkPosX = (int) (chunkPos & Integer.MAX_VALUE);
|
|
int chunkPosZ = (int) (chunkPos >> 32 & Integer.MAX_VALUE);
|
|
for (int x = chunkPosX << 4; x <= (chunkPosX << 4) + 16; x++) for (int z = chunkPosZ << 4; z <= (chunkPosZ << 4) + 16; z++) {
|
|
double distance = Math.hypot(x - posX, z - posZ);
|
|
if (distance <= getScale()) stomp(x, z, distance * 100 / getScale());
|
|
}
|
|
} else setDead();
|
|
}
|
|
|
|
tickDelay--;
|
|
|
|
if(this.isDead) {
|
|
if(RadiationConfig.rain > 0 && getScale() > 150) {
|
|
worldObj.getWorldInfo().setRaining(true);
|
|
worldObj.getWorldInfo().setThundering(true);
|
|
worldObj.getWorldInfo().setRainTime(RadiationConfig.rain);
|
|
worldObj.getWorldInfo().setThunderTime(RadiationConfig.rain);
|
|
AuxSavedData.setThunder(worldObj, RadiationConfig.rain);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private final List<Long> chunksToProcess = new ArrayList<>();
|
|
private final List<Long> outerChunksToProcess = new ArrayList<>();
|
|
|
|
// Is it worth the effort to split this into a method that can be called over multiple ticks? I'd say it's fast enough anyway...
|
|
private void gatherChunks() {
|
|
Set<Long> chunks = new LinkedHashSet<>(); // LinkedHashSet preserves insertion order
|
|
Set<Long> outerChunks = new LinkedHashSet<>();
|
|
int outerRange = getScale();
|
|
// Basically defines something like the step size, but as indirect proportion. The actual angle used for rotation will always end up at 360° for angle == adjustedMaxAngle
|
|
// So yea, I mathematically worked out that 20 is a good value for this, with the minimum possible being 18 in order to reach all chunks
|
|
int adjustedMaxAngle = 20 * outerRange / 32; // step size = 20 * chunks / 2
|
|
for (int angle = 0; angle <= adjustedMaxAngle; angle++) {
|
|
Vec3 vector = Vec3.createVectorHelper(outerRange, 0, 0);
|
|
vector.rotateAroundY((float) (angle * Math.PI / 180.0 / (adjustedMaxAngle / 360.0))); // Ugh, mutable data classes (also, ugh, radians; it uses degrees in 1.18; took me two hours to debug)
|
|
outerChunks.add(ChunkCoordIntPair.chunkXZ2Int((int) (posX + vector.xCoord) >> 4, (int) (posZ + vector.zCoord) >> 4));
|
|
}
|
|
for (int distance = 0; distance <= outerRange; distance += 8) for (int angle = 0; angle <= adjustedMaxAngle; angle++) {
|
|
Vec3 vector = Vec3.createVectorHelper(distance, 0, 0);
|
|
vector.rotateAroundY((float) (angle * Math.PI / 180.0 / (adjustedMaxAngle / 360.0)));
|
|
long chunkCoord = ChunkCoordIntPair.chunkXZ2Int((int) (posX + vector.xCoord) >> 4, (int) (posZ + vector.zCoord) >> 4);
|
|
if (!outerChunks.contains(chunkCoord)) chunks.add(chunkCoord);
|
|
}
|
|
|
|
chunksToProcess.addAll(chunks);
|
|
outerChunksToProcess.addAll(outerChunks);
|
|
Collections.reverse(chunksToProcess); // So it starts nicely from the middle
|
|
Collections.reverse(outerChunksToProcess);
|
|
}
|
|
|
|
// TODO cache chunks?
|
|
private void stomp(int x, int z, double dist) {
|
|
|
|
int depth = 0;
|
|
|
|
for(int y = 255; y >= 0; y--) {
|
|
|
|
Block b = worldObj.getBlock(x, y, z);
|
|
Block ab = worldObj.getBlock(x, y + 1, z);
|
|
int meta = worldObj.getBlockMetadata(x, y, z);
|
|
|
|
if(b.getMaterial() == Material.air)
|
|
continue;
|
|
|
|
if(b != ModBlocks.fallout && (ab == Blocks.air || (ab.isReplaceable(worldObj, x, y + 1, z) && !ab.getMaterial().isLiquid()))) {
|
|
|
|
double d = dist / 100;
|
|
|
|
double chance = 0.05 - Math.pow((d - 0.6) * 0.5, 2);
|
|
|
|
if(chance >= rand.nextDouble() && ModBlocks.fallout.canPlaceBlockAt(worldObj, x, y + 1, z))
|
|
worldObj.setBlock(x, y + 1, z, ModBlocks.fallout);
|
|
}
|
|
|
|
if(b.isFlammable(worldObj, x, y, z, ForgeDirection.UP)) {
|
|
if(rand.nextInt(5) == 0)
|
|
worldObj.setBlock(x, y + 1, z, Blocks.fire);
|
|
}
|
|
|
|
if (b == Blocks.leaves || b == Blocks.leaves2) {
|
|
worldObj.setBlock(x, y, z, Blocks.air);
|
|
}
|
|
|
|
else if(b == Blocks.stone) {
|
|
|
|
depth++;
|
|
|
|
if(dist < 5)
|
|
worldObj.setBlock(x, y, z, ModBlocks.sellafield_1);
|
|
else if(dist < 15)
|
|
worldObj.setBlock(x, y, z, ModBlocks.sellafield_0);
|
|
else if(dist < 75)
|
|
worldObj.setBlock(x, y, z, ModBlocks.sellafield_slaked);
|
|
else
|
|
return;
|
|
|
|
if(depth > 2)
|
|
return;
|
|
|
|
}else if(b == Blocks.grass) {
|
|
worldObj.setBlock(x, y, z, ModBlocks.waste_earth);
|
|
return;
|
|
|
|
} else if(b == Blocks.mycelium) {
|
|
worldObj.setBlock(x, y, z, ModBlocks.waste_mycelium);
|
|
return;
|
|
} else if(b == Blocks.sand) {
|
|
|
|
if(rand.nextInt(20) == 0)
|
|
worldObj.setBlock(x, y, z, meta == 0 ? ModBlocks.waste_trinitite : ModBlocks.waste_trinitite_red);
|
|
return;
|
|
}
|
|
|
|
else if (b == Blocks.clay) {
|
|
worldObj.setBlock(x, y, z, Blocks.hardened_clay);
|
|
return;
|
|
}
|
|
|
|
else if (b == Blocks.mossy_cobblestone) {
|
|
worldObj.setBlock(x, y, z, Blocks.coal_ore);
|
|
return;
|
|
}
|
|
|
|
else if (b == Blocks.coal_ore) {
|
|
int ra = rand.nextInt(150);
|
|
if (ra < 20) {
|
|
worldObj.setBlock(x, y, z, Blocks.diamond_ore);
|
|
} else if (ra < 30) {
|
|
worldObj.setBlock(x, y, z, Blocks.emerald_ore);
|
|
}
|
|
return;
|
|
}
|
|
|
|
else if (b == Blocks.log || b == Blocks.log2) {
|
|
worldObj.setBlock(x, y, z, ModBlocks.waste_log);
|
|
}
|
|
|
|
else if (b == Blocks.brown_mushroom_block || b == Blocks.red_mushroom_block) {
|
|
if (meta == 10) {
|
|
worldObj.setBlock(x, y, z, ModBlocks.waste_log);
|
|
} else {
|
|
worldObj.setBlock(x, y, z, Blocks.air,0,2);
|
|
}
|
|
}
|
|
|
|
else if (b.getMaterial() == Material.wood && b.isOpaqueCube() && b != ModBlocks.waste_log) {
|
|
worldObj.setBlock(x, y, z, ModBlocks.waste_planks);
|
|
}
|
|
|
|
else if (b == ModBlocks.ore_uranium) {
|
|
if (rand.nextInt(VersatileConfig.getSchrabOreChance()) == 0)
|
|
worldObj.setBlock(x, y, z, ModBlocks.ore_schrabidium);
|
|
else
|
|
worldObj.setBlock(x, y, z, ModBlocks.ore_uranium_scorched);
|
|
return;
|
|
}
|
|
|
|
else if (b == ModBlocks.ore_nether_uranium) {
|
|
if (rand.nextInt(VersatileConfig.getSchrabOreChance()) == 0)
|
|
worldObj.setBlock(x, y, z, ModBlocks.ore_nether_schrabidium);
|
|
else
|
|
worldObj.setBlock(x, y, z, ModBlocks.ore_nether_uranium_scorched);
|
|
return;
|
|
}
|
|
|
|
else if(b == ModBlocks.ore_gneiss_uranium) {
|
|
if(rand.nextInt(VersatileConfig.getSchrabOreChance()) == 0)
|
|
worldObj.setBlock(x, y, z, ModBlocks.ore_gneiss_schrabidium);
|
|
else
|
|
worldObj.setBlock(x, y, z, ModBlocks.ore_gneiss_uranium_scorched);
|
|
return;
|
|
|
|
//this piece stops the "stomp" from reaching below ground
|
|
} else if(b.isNormalCube()) {
|
|
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
@Override
|
|
protected void entityInit() {
|
|
this.dataWatcher.addObject(16, 0);
|
|
}
|
|
|
|
@Override
|
|
protected void readEntityFromNBT(NBTTagCompound tag) {
|
|
setScale(tag.getInteger("scale"));
|
|
chunksToProcess.addAll(readChunksFromIntArray(tag.getIntArray("chunks")));
|
|
outerChunksToProcess.addAll(readChunksFromIntArray(tag.getIntArray("outerChunks")));
|
|
}
|
|
|
|
private Collection<Long> readChunksFromIntArray(int[] data) {
|
|
List<Long> coords = new ArrayList<>();
|
|
boolean firstPart = true;
|
|
int x = 0;
|
|
for (int coord : data) {
|
|
if (firstPart) x = coord;
|
|
else coords.add(ChunkCoordIntPair.chunkXZ2Int(x, coord));
|
|
firstPart = !firstPart;
|
|
}
|
|
return coords;
|
|
}
|
|
|
|
@Override
|
|
protected void writeEntityToNBT(NBTTagCompound tag) {
|
|
tag.setInteger("scale", getScale());
|
|
tag.setIntArray("chunks", writeChunksToIntArray(chunksToProcess));
|
|
tag.setIntArray("outerChunks", writeChunksToIntArray(outerChunksToProcess));
|
|
}
|
|
|
|
private int[] writeChunksToIntArray(List<Long> coords) {
|
|
int[] data = new int[coords.size() * 2];
|
|
for (int i = 0; i < coords.size(); i++) {
|
|
data[i * 2] = (int) (coords.get(i) & Integer.MAX_VALUE);
|
|
data[i * 2 + 1] = (int) (coords.get(i) >> 32 & Integer.MAX_VALUE);
|
|
}
|
|
return data;
|
|
}
|
|
|
|
public void setScale(int i) {
|
|
this.dataWatcher.updateObject(16, i);
|
|
}
|
|
|
|
public int getScale() {
|
|
int scale = this.dataWatcher.getWatchableObjectInt(16);
|
|
return scale == 0 ? 1 : scale;
|
|
}
|
|
}
|