Started on a new, more modular approach to controller-less multiblocks, allowing for the creation of the EIM

This commit is contained in:
Aidan Brady 2014-08-01 13:04:31 -04:00
parent 1cfe6fc352
commit fc7dd3e65a
8 changed files with 293 additions and 144 deletions

View file

@ -43,7 +43,7 @@ public class RenderSalinationController extends TileEntitySpecialRenderer
data.side = ForgeDirection.getOrientation(tileEntity.facing); data.side = ForgeDirection.getOrientation(tileEntity.facing);
bindTexture(MekanismRenderer.getBlocksTexture()); bindTexture(MekanismRenderer.getBlocksTexture());
if(data.height >= 1 && tileEntity.waterTank.getCapacity() > 0) if(data.height >= 1 && tileEntity.waterTank.getCapacity() > 0)
{ {
Coord4D renderLoc = tileEntity.getRenderLocation(); Coord4D renderLoc = tileEntity.getRenderLocation();

View file

@ -4,15 +4,11 @@ import java.util.ArrayList;
import java.util.HashMap; import java.util.HashMap;
import java.util.HashSet; import java.util.HashSet;
import java.util.LinkedList; import java.util.LinkedList;
import java.util.Map;
import java.util.Queue; import java.util.Queue;
import java.util.Random; import java.util.Random;
import mekanism.api.Coord4D; import mekanism.api.Coord4D;
import mekanism.common.tank.DynamicTankCache; import mekanism.common.multiblock.MultiblockManager;
import mekanism.common.tile.TileEntityDynamicTank;
import net.minecraft.item.ItemStack;
import net.minecraft.tileentity.TileEntity;
import net.minecraft.world.ChunkCoordIntPair; import net.minecraft.world.ChunkCoordIntPair;
import net.minecraft.world.World; import net.minecraft.world.World;
import cpw.mods.fml.common.eventhandler.SubscribeEvent; import cpw.mods.fml.common.eventhandler.SubscribeEvent;
@ -24,6 +20,8 @@ public class CommonWorldTickHandler
{ {
private static final long maximumDeltaTimeNanoSecs = 16000000; // 16 milliseconds private static final long maximumDeltaTimeNanoSecs = 16000000; // 16 milliseconds
public boolean loaded;
private HashMap<Integer, Queue<ChunkCoordIntPair>> chunkRegenMap; private HashMap<Integer, Queue<ChunkCoordIntPair>> chunkRegenMap;
public void addRegenChunk(int dimensionId, ChunkCoordIntPair chunkCoord) public void addRegenChunk(int dimensionId, ChunkCoordIntPair chunkCoord)
@ -58,71 +56,36 @@ public class CommonWorldTickHandler
@SubscribeEvent @SubscribeEvent
public void onTick(WorldTickEvent event) public void onTick(WorldTickEvent event)
{ {
if(event.side == Side.SERVER && event.phase == Phase.END) if(event.side == Side.SERVER)
{ {
tickEnd(event.world); if(event.phase == Phase.START)
{
tickStart(event.world);
}
else if(event.phase == Phase.END)
{
tickEnd(event.world);
}
}
}
public void tickStart(World world)
{
if(!world.isRemote)
{
if(!loaded)
{
loaded = true;
MultiblockManager.load(world);
}
} }
} }
public void tickEnd(World world) public void tickEnd(World world)
{ {
ArrayList<Integer> idsToKill = new ArrayList<Integer>();
HashMap<Integer, HashSet<Coord4D>> tilesToKill = new HashMap<Integer, HashSet<Coord4D>>();
if(!world.isRemote) if(!world.isRemote)
{ {
for(Map.Entry<Integer, DynamicTankCache> entry : Mekanism.dynamicInventories.entrySet()) MultiblockManager.tick(world);
{
int inventoryID = entry.getKey();
for(Coord4D obj : entry.getValue().locations)
{
if(obj.dimensionId == world.provider.dimensionId)
{
TileEntity tileEntity = obj.getTileEntity(world);
if(!(tileEntity instanceof TileEntityDynamicTank) || ((TileEntityDynamicTank)tileEntity).inventoryID != inventoryID)
{
if(!tilesToKill.containsKey(inventoryID))
{
tilesToKill.put(inventoryID, new HashSet<Coord4D>());
}
tilesToKill.get(inventoryID).add(obj);
}
}
}
if(entry.getValue().locations.isEmpty())
{
idsToKill.add(inventoryID);
}
}
for(Map.Entry<Integer, HashSet<Coord4D>> entry : tilesToKill.entrySet())
{
for(Coord4D obj : entry.getValue())
{
Mekanism.dynamicInventories.get(entry.getKey()).locations.remove(obj);
}
}
for(int inventoryID : idsToKill)
{
for(Coord4D obj : Mekanism.dynamicInventories.get(inventoryID).locations)
{
TileEntityDynamicTank dynamicTank = (TileEntityDynamicTank)obj.getTileEntity(world);
if(dynamicTank != null)
{
dynamicTank.cachedData = new DynamicTankCache();
dynamicTank.inventory = new ItemStack[2];
dynamicTank.inventoryID = -1;
}
}
Mekanism.dynamicInventories.remove(inventoryID);
}
if(chunkRegenMap == null) if(chunkRegenMap == null)
{ {

View file

@ -51,6 +51,7 @@ import mekanism.common.item.ItemBlockGasTank;
import mekanism.common.item.ItemBlockMachine; import mekanism.common.item.ItemBlockMachine;
import mekanism.common.item.ItemBlockOre; import mekanism.common.item.ItemBlockOre;
import mekanism.common.item.ItemBlockPlastic; import mekanism.common.item.ItemBlockPlastic;
import mekanism.common.multiblock.MultiblockManager;
import mekanism.common.multipart.MultipartMekanism; import mekanism.common.multipart.MultipartMekanism;
import mekanism.common.network.PacketDataRequest.DataRequestMessage; import mekanism.common.network.PacketDataRequest.DataRequestMessage;
import mekanism.common.network.PacketTransmitterUpdate.PacketType; import mekanism.common.network.PacketTransmitterUpdate.PacketType;
@ -71,7 +72,6 @@ import mekanism.common.util.MekanismUtils;
import mekanism.common.util.MekanismUtils.ResourceType; import mekanism.common.util.MekanismUtils.ResourceType;
import mekanism.common.voice.VoiceServerManager; import mekanism.common.voice.VoiceServerManager;
import mekanism.common.world.GenHandler; import mekanism.common.world.GenHandler;
import net.minecraft.entity.EnumCreatureType; import net.minecraft.entity.EnumCreatureType;
import net.minecraft.init.Blocks; import net.minecraft.init.Blocks;
import net.minecraft.init.Items; import net.minecraft.init.Items;
@ -157,8 +157,8 @@ public class Mekanism
/** Map of Teleporters */ /** Map of Teleporters */
public static Map<Teleporter.Code, ArrayList<Coord4D>> teleporters = new HashMap<Teleporter.Code, ArrayList<Coord4D>>(); public static Map<Teleporter.Code, ArrayList<Coord4D>> teleporters = new HashMap<Teleporter.Code, ArrayList<Coord4D>>();
/** A map containing references to all dynamic tank inventory caches. */ /** The Dynamic Tank multiblock manager */
public static Map<Integer, DynamicTankCache> dynamicInventories = new HashMap<Integer, DynamicTankCache>(); public static MultiblockManager tankManager = new MultiblockManager("dynamicTank");
/** Mekanism creative tab */ /** Mekanism creative tab */
public static CreativeTabMekanism tabMekanism = new CreativeTabMekanism(); public static CreativeTabMekanism tabMekanism = new CreativeTabMekanism();
@ -1066,13 +1066,13 @@ public class Mekanism
//Clear all cache data //Clear all cache data
teleporters.clear(); teleporters.clear();
dynamicInventories.clear();
ic2Registered.clear(); ic2Registered.clear();
jetpackOn.clear(); jetpackOn.clear();
gasmaskOn.clear(); gasmaskOn.clear();
activeVibrators.clear(); activeVibrators.clear();
worldTickHandler.resetRegenChunks(); worldTickHandler.resetRegenChunks();
MultiblockManager.reset();
TransporterManager.flowingStacks.clear(); TransporterManager.flowingStacks.clear();
} }

View file

@ -0,0 +1,257 @@
package mekanism.common.multiblock;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
import mekanism.api.Coord4D;
import mekanism.common.tank.DynamicTankCache;
import mekanism.common.tile.TileEntityDynamicTank;
import net.minecraft.item.ItemStack;
import net.minecraft.nbt.NBTTagCompound;
import net.minecraft.nbt.NBTTagList;
import net.minecraft.tileentity.TileEntity;
import net.minecraft.world.World;
import net.minecraft.world.WorldSavedData;
import net.minecraftforge.common.util.Constants.NBT;
public class MultiblockManager
{
private static Set<MultiblockManager> managers = new HashSet<MultiblockManager>();
public DataHandler dataHandler;
public String name;
/** A map containing references to all dynamic tank inventory caches. */
public Map<Integer, DynamicTankCache> inventories = new HashMap<Integer, DynamicTankCache>();
public MultiblockManager(String s)
{
name = s;
managers.add(this);
}
public void createOrLoad(World world)
{
if(dataHandler == null)
{
dataHandler = (DataHandler)world.perWorldStorage.loadData(DataHandler.class, name);
if(dataHandler == null)
{
dataHandler = new DataHandler();
world.perWorldStorage.setData(name, dataHandler);
}
}
}
/**
* Grabs an inventory from the world's caches, and removes all the world's references to it.
* @param world - world the cache is stored in
* @param id - inventory ID to pull
* @return correct Dynamic Tank inventory cache
*/
public DynamicTankCache pullInventory(World world, int id)
{
DynamicTankCache toReturn = inventories.get(id);
for(Coord4D obj : inventories.get(id).locations)
{
TileEntityDynamicTank tileEntity = (TileEntityDynamicTank)obj.getTileEntity(world);
if(tileEntity != null)
{
tileEntity.cachedData = new DynamicTankCache();
tileEntity.inventory = new ItemStack[2];
tileEntity.inventoryID = -1;
}
}
inventories.remove(id);
dataHandler.markDirty();
return toReturn;
}
/**
* Updates a dynamic tank cache with the defined inventory ID with the parameterized values.
* @param inventoryID - inventory ID of the dynamic tank
* @param cache - cache of the dynamic tank
* @param tileEntity - dynamic tank TileEntity
*/
public void updateCache(int inventoryID, DynamicTankCache cache, TileEntityDynamicTank tileEntity)
{
if(!inventories.containsKey(inventoryID))
{
cache.locations.add(Coord4D.get(tileEntity));
inventories.put(inventoryID, cache);
return;
}
inventories.put(inventoryID, cache);
inventories.get(inventoryID).locations.add(Coord4D.get(tileEntity));
dataHandler.markDirty();
}
/**
* Grabs a unique inventory ID for a dynamic tank.
* @return unique inventory ID
*/
public int getUniqueInventoryID()
{
int id = 0;
while(true)
{
for(Integer i : inventories.keySet())
{
if(id == i)
{
id++;
continue;
}
}
return id;
}
}
public static void tick(World world)
{
for(MultiblockManager manager : managers)
{
ArrayList<Integer> idsToKill = new ArrayList<Integer>();
HashMap<Integer, HashSet<Coord4D>> tilesToKill = new HashMap<Integer, HashSet<Coord4D>>();
for(Map.Entry<Integer, DynamicTankCache> entry : manager.inventories.entrySet())
{
int inventoryID = entry.getKey();
for(Coord4D obj : entry.getValue().locations)
{
if(obj.dimensionId == world.provider.dimensionId)
{
TileEntity tileEntity = obj.getTileEntity(world);
if(!(tileEntity instanceof TileEntityDynamicTank) || ((TileEntityDynamicTank)tileEntity).inventoryID != inventoryID)
{
if(!tilesToKill.containsKey(inventoryID))
{
tilesToKill.put(inventoryID, new HashSet<Coord4D>());
}
tilesToKill.get(inventoryID).add(obj);
}
}
}
if(entry.getValue().locations.isEmpty())
{
idsToKill.add(inventoryID);
}
}
for(Map.Entry<Integer, HashSet<Coord4D>> entry : tilesToKill.entrySet())
{
for(Coord4D obj : entry.getValue())
{
manager.inventories.get(entry.getKey()).locations.remove(obj);
manager.dataHandler.markDirty();
}
}
for(int inventoryID : idsToKill)
{
for(Coord4D obj : manager.inventories.get(inventoryID).locations)
{
TileEntityDynamicTank dynamicTank = (TileEntityDynamicTank)obj.getTileEntity(world);
if(dynamicTank != null)
{
dynamicTank.cachedData = new DynamicTankCache();
dynamicTank.inventory = new ItemStack[2];
dynamicTank.inventoryID = -1;
}
}
manager.inventories.remove(inventoryID);
manager.dataHandler.markDirty();
}
}
}
public static void load(World world)
{
for(MultiblockManager manager : managers)
{
manager.createOrLoad(world);
}
}
public static void reset()
{
for(MultiblockManager manager : managers)
{
manager.inventories.clear();
manager.dataHandler = null;
}
}
public class DataHandler extends WorldSavedData
{
public DataHandler()
{
super(name);
}
@Override
public void readFromNBT(NBTTagCompound nbtTags)
{
NBTTagList list = nbtTags.getTagList("invList", NBT.TAG_COMPOUND);
for(int i = 0; i < list.tagCount(); i++)
{
NBTTagCompound compound = list.getCompoundTagAt(i);
DynamicTankCache cache = new DynamicTankCache();
cache.load(compound);
NBTTagList coordsList = compound.getTagList("coordsList", NBT.TAG_COMPOUND);
for(int j = 0; j < coordsList.tagCount(); j++)
{
cache.locations.add(Coord4D.read(coordsList.getCompoundTagAt(j)));
}
}
}
@Override
public void writeToNBT(NBTTagCompound nbtTags)
{
NBTTagList list = new NBTTagList();
for(Map.Entry<Integer, DynamicTankCache> entry : inventories.entrySet())
{
NBTTagCompound compound = new NBTTagCompound();
compound.setInteger("id", entry.getKey());
entry.getValue().save(compound);
NBTTagList coordsList = new NBTTagList();
for(Coord4D coord : entry.getValue().locations)
{
coordsList.appendTag(coord.write(new NBTTagCompound()));
}
compound.setTag("coordsList", coordsList);
list.appendTag(compound);
}
nbtTags.setTag("invList", list);
}
}
}

View file

@ -400,16 +400,16 @@ public class TankUpdateProtocol
{ {
for(int id : idsFound) for(int id : idsFound)
{ {
if(Mekanism.dynamicInventories.get(id) != null) if(Mekanism.tankManager.inventories.get(id) != null)
{ {
cache = MekanismUtils.pullInventory(pointer.getWorldObj(), id); cache = Mekanism.tankManager.pullInventory(pointer.getWorldObj(), id);
idToUse = id; idToUse = id;
break; break;
} }
} }
} }
else { else {
idToUse = MekanismUtils.getUniqueInventoryID(); idToUse = Mekanism.tankManager.getUniqueInventoryID();
} }
cache.apply(structureFound); cache.apply(structureFound);

View file

@ -151,7 +151,7 @@ public class TileEntityDynamicTank extends TileEntityContainerBlock implements I
if(inventoryID != -1 && structure == null) if(inventoryID != -1 && structure == null)
{ {
MekanismUtils.updateCache(inventoryID, cachedData, this); Mekanism.tankManager.updateCache(inventoryID, cachedData, this);
} }
if(structure == null && ticker == 5) if(structure == null && ticker == 5)
@ -190,7 +190,7 @@ public class TileEntityDynamicTank extends TileEntityContainerBlock implements I
if(inventoryID != -1) if(inventoryID != -1)
{ {
cachedData.sync(structure); cachedData.sync(structure);
MekanismUtils.updateCache(inventoryID, cachedData, this); Mekanism.tankManager.updateCache(inventoryID, cachedData, this);
} }
manageInventory(); manageInventory();

View file

@ -486,7 +486,7 @@ public class TileEntityLogisticalSorter extends TileEntityElectricBlock implemen
return new int[] {0}; return new int[] {0};
} }
return null; return InventoryUtils.EMPTY;
} }
@Override @Override

View file

@ -887,77 +887,6 @@ public final class MekanismUtils
player.openContainer.addCraftingToCrafters(player); player.openContainer.addCraftingToCrafters(player);
} }
/**
* Grabs an inventory from the world's caches, and removes all the world's references to it.
* @param world - world the cache is stored in
* @param id - inventory ID to pull
* @return correct Dynamic Tank inventory cache
*/
public static DynamicTankCache pullInventory(World world, int id)
{
DynamicTankCache toReturn = Mekanism.dynamicInventories.get(id);
for(Coord4D obj : Mekanism.dynamicInventories.get(id).locations)
{
TileEntityDynamicTank tileEntity = (TileEntityDynamicTank)obj.getTileEntity(world);
if(tileEntity != null)
{
tileEntity.cachedData = new DynamicTankCache();
tileEntity.inventory = new ItemStack[2];
tileEntity.inventoryID = -1;
}
}
Mekanism.dynamicInventories.remove(id);
return toReturn;
}
/**
* Updates a dynamic tank cache with the defined inventory ID with the parameterized values.
* @param inventoryID - inventory ID of the dynamic tank
* @param cache - cache of the dynamic tank
* @param tileEntity - dynamic tank TileEntity
*/
public static void updateCache(int inventoryID, DynamicTankCache cache, TileEntityDynamicTank tileEntity)
{
if(!Mekanism.dynamicInventories.containsKey(inventoryID))
{
cache.locations.add(Coord4D.get(tileEntity));
Mekanism.dynamicInventories.put(inventoryID, cache);
return;
}
Mekanism.dynamicInventories.put(inventoryID, cache);
Mekanism.dynamicInventories.get(inventoryID).locations.add(Coord4D.get(tileEntity));
}
/**
* Grabs a unique inventory ID for a dynamic tank.
* @return unique inventory ID
*/
public static int getUniqueInventoryID()
{
int id = 0;
while(true)
{
for(Integer i : Mekanism.dynamicInventories.keySet())
{
if(id == i)
{
id++;
continue;
}
}
return id;
}
}
/** /**
* Retrieves a private value from a defined class and field. * Retrieves a private value from a defined class and field.
* @param obj - the Object to retrieve the value from, null if static * @param obj - the Object to retrieve the value from, null if static