519 lines
17 KiB
Java
519 lines
17 KiB
Java
package com.pahimar.ee3.tileentity;
|
|
|
|
import com.pahimar.ee3.network.PacketHandler;
|
|
import com.pahimar.ee3.network.message.MessageTileCalcinator;
|
|
import com.pahimar.ee3.reference.Names;
|
|
import com.pahimar.ee3.util.CalcinationHelper;
|
|
import cpw.mods.fml.relauncher.Side;
|
|
import cpw.mods.fml.relauncher.SideOnly;
|
|
import net.minecraft.entity.item.EntityItem;
|
|
import net.minecraft.entity.player.EntityPlayer;
|
|
import net.minecraft.inventory.ISidedInventory;
|
|
import net.minecraft.item.ItemStack;
|
|
import net.minecraft.nbt.NBTTagCompound;
|
|
import net.minecraft.nbt.NBTTagList;
|
|
import net.minecraft.network.Packet;
|
|
import net.minecraft.tileentity.TileEntityFurnace;
|
|
import net.minecraft.tileentity.TileEntityHopper;
|
|
import net.minecraftforge.common.util.ForgeDirection;
|
|
|
|
public class TileEntityCalcinator extends TileEntityEE implements ISidedInventory
|
|
{
|
|
public static final int INVENTORY_SIZE = 4;
|
|
public static final int FUEL_INVENTORY_INDEX = 0;
|
|
public static final int INPUT_INVENTORY_INDEX = 1;
|
|
public static final int OUTPUT_LEFT_INVENTORY_INDEX = 2;
|
|
public static final int OUTPUT_RIGHT_INVENTORY_INDEX = 3;
|
|
private static final int DEFAULT_ITEM_SUCK_COOL_DOWN = 20;
|
|
public int deviceCookTime; // How much longer the Calcinator will cook
|
|
public int fuelBurnTime; // The fuel value for the currently burning fuel
|
|
public int itemCookTime; // How long the current item has been "cooking"
|
|
|
|
public byte leftStackSize, leftStackMeta, rightStackSize, rightStackMeta;
|
|
|
|
public int itemSuckCoolDown = 0;
|
|
/**
|
|
* The ItemStacks that hold the items currently being used in the Calcinator
|
|
*/
|
|
private ItemStack[] inventory;
|
|
|
|
public TileEntityCalcinator()
|
|
{
|
|
inventory = new ItemStack[INVENTORY_SIZE];
|
|
}
|
|
|
|
public static boolean suckInItems(TileEntityCalcinator tileEntityCalcinator)
|
|
{
|
|
EntityItem entityitem = TileEntityHopper.func_145897_a(tileEntityCalcinator.getWorldObj(), tileEntityCalcinator.xCoord, tileEntityCalcinator.yCoord + 1.0D, tileEntityCalcinator.zCoord);
|
|
|
|
return entityitem != null && TileEntityHopper.func_145898_a(tileEntityCalcinator, entityitem);
|
|
}
|
|
|
|
@Override
|
|
public int[] getAccessibleSlotsFromSide(int side)
|
|
{
|
|
return side == ForgeDirection.DOWN.ordinal() ? new int[]{FUEL_INVENTORY_INDEX, OUTPUT_LEFT_INVENTORY_INDEX, OUTPUT_RIGHT_INVENTORY_INDEX} : new int[]{FUEL_INVENTORY_INDEX, INPUT_INVENTORY_INDEX, OUTPUT_LEFT_INVENTORY_INDEX, OUTPUT_RIGHT_INVENTORY_INDEX};
|
|
}
|
|
|
|
@Override
|
|
public boolean canInsertItem(int slotIndex, ItemStack itemStack, int side) {
|
|
|
|
return isItemValidForSlot(slotIndex, itemStack);
|
|
}
|
|
|
|
@Override
|
|
public boolean canExtractItem(int slotIndex, ItemStack itemStack, int side)
|
|
{
|
|
return slotIndex == OUTPUT_LEFT_INVENTORY_INDEX || slotIndex == OUTPUT_RIGHT_INVENTORY_INDEX;
|
|
}
|
|
|
|
@Override
|
|
public void readFromNBT(NBTTagCompound nbtTagCompound)
|
|
{
|
|
super.readFromNBT(nbtTagCompound);
|
|
|
|
// Read in the ItemStacks in the inventory from NBT
|
|
NBTTagList tagList = nbtTagCompound.getTagList(Names.NBT.ITEMS, 10);
|
|
inventory = new ItemStack[this.getSizeInventory()];
|
|
for (int i = 0; i < tagList.tagCount(); ++i)
|
|
{
|
|
NBTTagCompound tagCompound = tagList.getCompoundTagAt(i);
|
|
byte slotIndex = tagCompound.getByte("Slot");
|
|
if (slotIndex >= 0 && slotIndex < inventory.length)
|
|
{
|
|
inventory[slotIndex] = ItemStack.loadItemStackFromNBT(tagCompound);
|
|
}
|
|
}
|
|
|
|
deviceCookTime = nbtTagCompound.getInteger("deviceCookTime");
|
|
fuelBurnTime = nbtTagCompound.getInteger("fuelBurnTime");
|
|
itemCookTime = nbtTagCompound.getInteger("itemCookTime");
|
|
itemSuckCoolDown = nbtTagCompound.getInteger("itemSuckCoolDown");
|
|
}
|
|
|
|
@Override
|
|
public int getSizeInventory()
|
|
{
|
|
return inventory.length;
|
|
}
|
|
|
|
@Override
|
|
public ItemStack getStackInSlot(int slotIndex)
|
|
{
|
|
sendDustPileData();
|
|
return inventory[slotIndex];
|
|
}
|
|
|
|
@Override
|
|
public ItemStack decrStackSize(int slotIndex, int decrementAmount)
|
|
{
|
|
ItemStack itemStack = getStackInSlot(slotIndex);
|
|
if (itemStack != null)
|
|
{
|
|
if (itemStack.stackSize <= decrementAmount)
|
|
{
|
|
setInventorySlotContents(slotIndex, null);
|
|
}
|
|
else
|
|
{
|
|
itemStack = itemStack.splitStack(decrementAmount);
|
|
if (itemStack.stackSize == 0)
|
|
{
|
|
setInventorySlotContents(slotIndex, null);
|
|
}
|
|
}
|
|
}
|
|
|
|
return itemStack;
|
|
}
|
|
|
|
@Override
|
|
public ItemStack getStackInSlotOnClosing(int slotIndex)
|
|
{
|
|
ItemStack itemStack = getStackInSlot(slotIndex);
|
|
if (itemStack != null)
|
|
{
|
|
setInventorySlotContents(slotIndex, null);
|
|
}
|
|
return itemStack;
|
|
}
|
|
|
|
@Override
|
|
public void setInventorySlotContents(int slotIndex, ItemStack itemStack)
|
|
{
|
|
inventory[slotIndex] = itemStack;
|
|
if (itemStack != null && itemStack.stackSize > getInventoryStackLimit())
|
|
{
|
|
itemStack.stackSize = getInventoryStackLimit();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public String getInventoryName()
|
|
{
|
|
return this.hasCustomName() ? this.getCustomName() : Names.Containers.CALCINATOR;
|
|
}
|
|
|
|
@Override
|
|
public boolean hasCustomInventoryName()
|
|
{
|
|
return this.hasCustomName();
|
|
}
|
|
|
|
@Override
|
|
public int getInventoryStackLimit()
|
|
{
|
|
return 64;
|
|
}
|
|
|
|
@Override
|
|
public boolean isUseableByPlayer(EntityPlayer entityplayer)
|
|
{
|
|
return this.worldObj.getTileEntity(xCoord, yCoord, zCoord) == this && entityplayer.getDistanceSq((double) xCoord + 0.5D, (double) yCoord + 0.5D, (double) zCoord + 0.5D) <= 64D;
|
|
}
|
|
|
|
@Override
|
|
public void openInventory()
|
|
{
|
|
// NOOP
|
|
}
|
|
|
|
@Override
|
|
public void closeInventory()
|
|
{
|
|
// NOOP
|
|
}
|
|
|
|
@Override
|
|
public boolean isItemValidForSlot(int slotIndex, ItemStack itemStack) {
|
|
|
|
if (itemStack != null) {
|
|
if (slotIndex == FUEL_INVENTORY_INDEX) {
|
|
return TileEntityFurnace.isItemFuel(itemStack);
|
|
}
|
|
else if (slotIndex == INPUT_INVENTORY_INDEX) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
private void sendDustPileData()
|
|
{
|
|
if (this.getBlockType() != null)
|
|
{
|
|
this.worldObj.addBlockEvent(this.xCoord, this.yCoord, this.zCoord, this.getBlockType(), 2, getLeftStackSize());
|
|
this.worldObj.addBlockEvent(this.xCoord, this.yCoord, this.zCoord, this.getBlockType(), 3, getLeftStackMeta());
|
|
this.worldObj.addBlockEvent(this.xCoord, this.yCoord, this.zCoord, this.getBlockType(), 4, getRightStackSize());
|
|
this.worldObj.addBlockEvent(this.xCoord, this.yCoord, this.zCoord, this.getBlockType(), 5, getRightStackMeta());
|
|
}
|
|
}
|
|
|
|
private int getLeftStackSize()
|
|
{
|
|
if (this.inventory[OUTPUT_LEFT_INVENTORY_INDEX] != null)
|
|
{
|
|
return this.inventory[OUTPUT_LEFT_INVENTORY_INDEX].stackSize;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
private int getLeftStackMeta()
|
|
{
|
|
if (this.inventory[OUTPUT_LEFT_INVENTORY_INDEX] != null)
|
|
{
|
|
return this.inventory[OUTPUT_LEFT_INVENTORY_INDEX].getItemDamage();
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
private int getRightStackSize()
|
|
{
|
|
if (this.inventory[OUTPUT_RIGHT_INVENTORY_INDEX] != null)
|
|
{
|
|
return this.inventory[OUTPUT_RIGHT_INVENTORY_INDEX].stackSize;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
private int getRightStackMeta()
|
|
{
|
|
if (this.inventory[OUTPUT_RIGHT_INVENTORY_INDEX] != null)
|
|
{
|
|
return this.inventory[OUTPUT_RIGHT_INVENTORY_INDEX].getItemDamage();
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
@Override
|
|
public void writeToNBT(NBTTagCompound nbtTagCompound)
|
|
{
|
|
super.writeToNBT(nbtTagCompound);
|
|
|
|
// Write the ItemStacks in the inventory to NBT
|
|
NBTTagList tagList = new NBTTagList();
|
|
for (int currentIndex = 0; currentIndex < inventory.length; ++currentIndex)
|
|
{
|
|
if (inventory[currentIndex] != null)
|
|
{
|
|
NBTTagCompound tagCompound = new NBTTagCompound();
|
|
tagCompound.setByte("Slot", (byte) currentIndex);
|
|
inventory[currentIndex].writeToNBT(tagCompound);
|
|
tagList.appendTag(tagCompound);
|
|
}
|
|
}
|
|
nbtTagCompound.setTag(Names.NBT.ITEMS, tagList);
|
|
nbtTagCompound.setInteger("deviceCookTime", deviceCookTime);
|
|
nbtTagCompound.setInteger("fuelBurnTime", fuelBurnTime);
|
|
nbtTagCompound.setInteger("itemCookTime", itemCookTime);
|
|
nbtTagCompound.setInteger("itemSuckCoolDown", itemSuckCoolDown);
|
|
}
|
|
|
|
@Override
|
|
public Packet getDescriptionPacket()
|
|
{
|
|
sendDustPileData();
|
|
return PacketHandler.INSTANCE.getPacketFrom(new MessageTileCalcinator(this));
|
|
}
|
|
|
|
@SideOnly(Side.CLIENT)
|
|
public int getCookProgressScaled(int scale)
|
|
{
|
|
return this.itemCookTime * scale / 200;
|
|
}
|
|
|
|
@SideOnly(Side.CLIENT)
|
|
public int getBurnTimeRemainingScaled(int scale)
|
|
{
|
|
if (this.fuelBurnTime > 0)
|
|
{
|
|
return this.deviceCookTime * scale / this.fuelBurnTime;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
@Override
|
|
public void updateEntity()
|
|
{
|
|
boolean isBurning = this.deviceCookTime > 0;
|
|
boolean sendUpdate = false;
|
|
|
|
// If the Calcinator still has burn time, decrement it
|
|
if (this.deviceCookTime > 0)
|
|
{
|
|
this.deviceCookTime--;
|
|
}
|
|
|
|
if (!this.worldObj.isRemote)
|
|
{
|
|
// Start "cooking" a new item, if we can
|
|
if (this.deviceCookTime == 0 && this.canCalcinate())
|
|
{
|
|
this.fuelBurnTime = this.deviceCookTime = TileEntityFurnace.getItemBurnTime(this.inventory[FUEL_INVENTORY_INDEX]);
|
|
|
|
if (this.deviceCookTime > 0)
|
|
{
|
|
sendUpdate = true;
|
|
|
|
if (this.inventory[FUEL_INVENTORY_INDEX] != null)
|
|
{
|
|
--this.inventory[FUEL_INVENTORY_INDEX].stackSize;
|
|
|
|
if (this.inventory[FUEL_INVENTORY_INDEX].stackSize == 0)
|
|
{
|
|
this.inventory[FUEL_INVENTORY_INDEX] = this.inventory[FUEL_INVENTORY_INDEX].getItem().getContainerItem(inventory[FUEL_INVENTORY_INDEX]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Continue "cooking" the same item, if we can
|
|
if (this.deviceCookTime > 0 && this.canCalcinate())
|
|
{
|
|
this.itemCookTime++;
|
|
|
|
if (this.itemCookTime == 200)
|
|
{
|
|
this.itemCookTime = 0;
|
|
this.calcinateItem();
|
|
sendUpdate = true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
this.itemCookTime = 0;
|
|
}
|
|
|
|
// If the state has changed, catch that something changed
|
|
if (isBurning != this.deviceCookTime > 0)
|
|
{
|
|
sendUpdate = true;
|
|
}
|
|
|
|
//Item sucking
|
|
if (this.itemSuckCoolDown > 0)
|
|
{
|
|
itemSuckCoolDown--;
|
|
}
|
|
else
|
|
{
|
|
if (suckInItems(this))
|
|
{
|
|
markDirty();
|
|
}
|
|
itemSuckCoolDown = DEFAULT_ITEM_SUCK_COOL_DOWN;
|
|
}
|
|
}
|
|
|
|
if (sendUpdate)
|
|
{
|
|
this.markDirty();
|
|
this.state = this.deviceCookTime > 0 ? (byte) 1 : (byte) 0;
|
|
this.worldObj.addBlockEvent(this.xCoord, this.yCoord, this.zCoord, this.getBlockType(), 1, this.state);
|
|
sendDustPileData();
|
|
this.worldObj.notifyBlockChange(this.xCoord, this.yCoord, this.zCoord, this.getBlockType());
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public boolean receiveClientEvent(int eventId, int eventData)
|
|
{
|
|
if (eventId == 1)
|
|
{
|
|
this.state = (byte) eventData;
|
|
// NAME UPDATE - this.worldObj.updateAllLightTypes(this.xCoord, this.yCoord, this.zCoord);
|
|
this.worldObj.func_147451_t(this.xCoord, this.yCoord, this.zCoord);
|
|
return true;
|
|
}
|
|
else if (eventId == 2)
|
|
{
|
|
this.leftStackSize = (byte) eventData;
|
|
return true;
|
|
}
|
|
else if (eventId == 3)
|
|
{
|
|
this.leftStackMeta = (byte) eventData;
|
|
return true;
|
|
}
|
|
else if (eventId == 4)
|
|
{
|
|
this.rightStackSize = (byte) eventData;
|
|
return true;
|
|
}
|
|
else if (eventId == 5)
|
|
{
|
|
this.rightStackMeta = (byte) eventData;
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
return super.receiveClientEvent(eventId, eventData);
|
|
}
|
|
}
|
|
|
|
private boolean canCalcinate()
|
|
{
|
|
if (inventory[INPUT_INVENTORY_INDEX] == null)
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
ItemStack alchemicalDustStack = CalcinationHelper.getCalcinationResult(this.inventory[INPUT_INVENTORY_INDEX]);
|
|
|
|
/**
|
|
* If we don't get a calcination result, then return false
|
|
*/
|
|
if (alchemicalDustStack == null)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* If either slot is empty, return true (we have a valid calcination result
|
|
*/
|
|
if (this.inventory[OUTPUT_LEFT_INVENTORY_INDEX] == null || this.inventory[OUTPUT_RIGHT_INVENTORY_INDEX] == null)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
boolean leftEquals = this.inventory[OUTPUT_LEFT_INVENTORY_INDEX].isItemEqual(alchemicalDustStack);
|
|
int leftResult = this.inventory[OUTPUT_LEFT_INVENTORY_INDEX].stackSize + alchemicalDustStack.stackSize;
|
|
|
|
boolean rightEquals = this.inventory[OUTPUT_RIGHT_INVENTORY_INDEX].isItemEqual(alchemicalDustStack);
|
|
int rightResult = this.inventory[OUTPUT_RIGHT_INVENTORY_INDEX].stackSize + alchemicalDustStack.stackSize;
|
|
|
|
if (!leftEquals && !rightEquals)
|
|
{
|
|
return false;
|
|
}
|
|
else if (leftEquals && !rightEquals)
|
|
{
|
|
return leftResult <= getInventoryStackLimit() && leftResult <= alchemicalDustStack.getMaxStackSize();
|
|
}
|
|
else if (!leftEquals)
|
|
{
|
|
return rightResult <= getInventoryStackLimit() && rightResult <= alchemicalDustStack.getMaxStackSize();
|
|
}
|
|
else
|
|
{
|
|
return leftResult <= getInventoryStackLimit() && leftResult <= alchemicalDustStack.getMaxStackSize() || rightResult <= getInventoryStackLimit() && rightResult <= alchemicalDustStack.getMaxStackSize();
|
|
}
|
|
}
|
|
}
|
|
|
|
public void calcinateItem()
|
|
{
|
|
if (this.canCalcinate())
|
|
{
|
|
ItemStack alchemicalDustStack = CalcinationHelper.getCalcinationResult(this.inventory[INPUT_INVENTORY_INDEX]);
|
|
addItemStackToOutput(alchemicalDustStack.copy());
|
|
|
|
this.inventory[INPUT_INVENTORY_INDEX].stackSize--;
|
|
|
|
if (this.inventory[INPUT_INVENTORY_INDEX].stackSize <= 0)
|
|
{
|
|
this.inventory[INPUT_INVENTORY_INDEX] = null;
|
|
}
|
|
}
|
|
}
|
|
|
|
private void addItemStackToOutput(ItemStack alchemicalDustStack)
|
|
{
|
|
int maxStackSize = Math.min(getInventoryStackLimit(), alchemicalDustStack.getMaxStackSize());
|
|
|
|
if (this.inventory[OUTPUT_LEFT_INVENTORY_INDEX] == null)
|
|
{
|
|
this.inventory[OUTPUT_LEFT_INVENTORY_INDEX] = alchemicalDustStack;
|
|
return;
|
|
}
|
|
if (this.inventory[OUTPUT_LEFT_INVENTORY_INDEX].isItemEqual(alchemicalDustStack) && this.inventory[OUTPUT_LEFT_INVENTORY_INDEX].stackSize < maxStackSize)
|
|
{
|
|
int addedSize = Math.min(alchemicalDustStack.stackSize, maxStackSize - this.inventory[OUTPUT_LEFT_INVENTORY_INDEX].stackSize);
|
|
alchemicalDustStack.stackSize -= addedSize;
|
|
this.inventory[OUTPUT_LEFT_INVENTORY_INDEX].stackSize += addedSize;
|
|
if (alchemicalDustStack.stackSize == 0)
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
if (this.inventory[OUTPUT_RIGHT_INVENTORY_INDEX] == null)
|
|
{
|
|
this.inventory[OUTPUT_RIGHT_INVENTORY_INDEX] = alchemicalDustStack;
|
|
return;
|
|
}
|
|
if (this.inventory[OUTPUT_RIGHT_INVENTORY_INDEX].isItemEqual(alchemicalDustStack) && this.inventory[OUTPUT_RIGHT_INVENTORY_INDEX].stackSize < maxStackSize)
|
|
{
|
|
int addedSize = Math.min(alchemicalDustStack.stackSize, maxStackSize - this.inventory[OUTPUT_RIGHT_INVENTORY_INDEX].stackSize);
|
|
alchemicalDustStack.stackSize -= addedSize;
|
|
this.inventory[OUTPUT_RIGHT_INVENTORY_INDEX].stackSize += addedSize;
|
|
}
|
|
}
|
|
}
|