equivalent-exchange-3/src/main/java/com/pahimar/ee3/tileentity/TileEntityCalcinator.java
2023-01-03 17:47:36 +01:00

505 lines
18 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;
}
}
}