BallOfEnergy e38ce02ccf Upgrade rework, now with 3463% more caching!
(it was slow and needed a redo)
2024-10-14 22:00:39 -05:00

370 lines
10 KiB
Java

package com.hbm.tileentity.machine;
import java.util.HashMap;
import java.util.List;
import com.hbm.blocks.ModBlocks;
import com.hbm.interfaces.IControlReceiver;
import com.hbm.inventory.UpgradeManagerNT;
import com.hbm.inventory.container.ContainerMixer;
import com.hbm.inventory.fluid.Fluids;
import com.hbm.inventory.fluid.tank.FluidTank;
import com.hbm.inventory.gui.GUIMixer;
import com.hbm.inventory.recipes.MixerRecipes;
import com.hbm.inventory.recipes.MixerRecipes.MixerRecipe;
import com.hbm.items.machine.ItemMachineUpgrade.UpgradeType;
import com.hbm.lib.Library;
import com.hbm.tileentity.*;
import com.hbm.util.BobMathUtil;
import com.hbm.util.I18nUtil;
import com.hbm.util.fauxpointtwelve.DirPos;
import api.hbm.energymk2.IEnergyReceiverMK2;
import api.hbm.fluid.IFluidStandardTransceiver;
import cpw.mods.fml.relauncher.Side;
import cpw.mods.fml.relauncher.SideOnly;
import io.netty.buffer.ByteBuf;
import net.minecraft.entity.player.EntityPlayer;
import net.minecraft.inventory.Container;
import net.minecraft.item.ItemStack;
import net.minecraft.nbt.NBTTagCompound;
import net.minecraft.util.AxisAlignedBB;
import net.minecraft.util.EnumChatFormatting;
import net.minecraft.world.World;
public class TileEntityMachineMixer extends TileEntityMachineBase implements IControlReceiver, IGUIProvider, IEnergyReceiverMK2, IFluidStandardTransceiver, IUpgradeInfoProvider, IFluidCopiable {
public long power;
public static final long maxPower = 10_000;
public int progress;
public int processTime;
public int recipeIndex;
public float rotation;
public float prevRotation;
public boolean wasOn = false;
private int consumption = 50;
public FluidTank[] tanks;
public UpgradeManagerNT upgradeManager = new UpgradeManagerNT();
public TileEntityMachineMixer() {
super(5);
this.tanks = new FluidTank[3];
this.tanks[0] = new FluidTank(Fluids.NONE, 16_000);
this.tanks[1] = new FluidTank(Fluids.NONE, 16_000);
this.tanks[2] = new FluidTank(Fluids.NONE, 24_000);
}
@Override
public String getName() {
return "container.machineMixer";
}
@Override
public void updateEntity() {
if(!worldObj.isRemote) {
this.power = Library.chargeTEFromItems(slots, 0, power, maxPower);
tanks[2].setType(2, slots);
upgradeManager.checkSlots(this, slots, 3, 4);
int speedLevel = upgradeManager.getLevel(UpgradeType.SPEED);
int powerLevel = upgradeManager.getLevel(UpgradeType.POWER);
int overLevel = upgradeManager.getLevel(UpgradeType.OVERDRIVE);
this.consumption = 50;
this.consumption += speedLevel * 150;
this.consumption -= this.consumption * powerLevel * 0.25;
this.consumption *= (overLevel * 3 + 1);
for(DirPos pos : getConPos()) {
this.trySubscribe(worldObj, pos.getX(), pos.getY(), pos.getZ(), pos.getDir());
if(tanks[0].getTankType() != Fluids.NONE) this.trySubscribe(tanks[0].getTankType(), worldObj, pos.getX(), pos.getY(), pos.getZ(), pos.getDir());
if(tanks[1].getTankType() != Fluids.NONE) this.trySubscribe(tanks[1].getTankType(), worldObj, pos.getX(), pos.getY(), pos.getZ(), pos.getDir());
}
this.wasOn = this.canProcess();
if(this.wasOn) {
this.progress++;
this.power -= this.getConsumption();
this.processTime -= this.processTime * speedLevel / 4;
this.processTime /= (overLevel + 1);
if(processTime <= 0) this.processTime = 1;
if(this.progress >= this.processTime) {
this.process();
this.progress = 0;
}
} else {
this.progress = 0;
}
for(DirPos pos : getConPos()) {
if(tanks[2].getFill() > 0) this.sendFluid(tanks[2], worldObj, pos.getX(), pos.getY(), pos.getZ(), pos.getDir());
}
NBTTagCompound data = new NBTTagCompound();
data.setLong("power", power);
data.setInteger("processTime", processTime);
data.setInteger("progress", progress);
data.setInteger("recipe", recipeIndex);
data.setBoolean("wasOn", wasOn);
for(int i = 0; i < 3; i++) {
tanks[i].writeToNBT(data, i + "");
}
this.networkPackNT(50);
} else {
this.prevRotation = this.rotation;
if(this.wasOn) {
this.rotation += 20F;
}
if(this.rotation >= 360) {
this.rotation -= 360;
this.prevRotation -= 360;
}
}
}
@Override
public void serialize(ByteBuf buf) {
super.serialize(buf);
buf.writeLong(power);
buf.writeInt(processTime);
buf.writeInt(progress);
buf.writeInt(recipeIndex);
buf.writeBoolean(wasOn);
for(int i = 0; i < tanks.length; i++) tanks[i].serialize(buf);
}
@Override
public void deserialize(ByteBuf buf) {
super.deserialize(buf);
power = buf.readLong();
processTime = buf.readInt();
progress = buf.readInt();
recipeIndex = buf.readInt();
wasOn = buf.readBoolean();
for(int i = 0; i < tanks.length; i++) tanks[i].deserialize(buf);
}
public boolean canProcess() {
MixerRecipe[] recipes = MixerRecipes.getOutput(tanks[2].getTankType());
if(recipes == null || recipes.length <= 0) {
this.recipeIndex = 0;
return false;
}
this.recipeIndex = this.recipeIndex % recipes.length;
MixerRecipe recipe = recipes[this.recipeIndex];
if(recipe == null) {
this.recipeIndex = 0;
return false;
}
tanks[0].setTankType(recipe.input1 != null ? recipe.input1.type : Fluids.NONE);
tanks[1].setTankType(recipe.input2 != null ? recipe.input2.type : Fluids.NONE);
if(recipe.input1 != null && tanks[0].getFill() < recipe.input1.fill) return false;
if(recipe.input2 != null && tanks[1].getFill() < recipe.input2.fill) return false;
/* simplest check would usually go first, but fluid checks also do the setup and we want that to happen even without power */
if(this.power < getConsumption()) return false;
if(recipe.output + tanks[2].getFill() > tanks[2].getMaxFill()) return false;
if(recipe.solidInput != null) {
if(slots[1] == null) return false;
if(!recipe.solidInput.matchesRecipe(slots[1], true) || recipe.solidInput.stacksize > slots[1].stackSize) return false;
}
this.processTime = recipe.processTime;
return true;
}
protected void process() {
MixerRecipe[] recipes = MixerRecipes.getOutput(tanks[2].getTankType());
MixerRecipe recipe = recipes[this.recipeIndex % recipes.length];
if(recipe.input1 != null) tanks[0].setFill(tanks[0].getFill() - recipe.input1.fill);
if(recipe.input2 != null) tanks[1].setFill(tanks[1].getFill() - recipe.input2.fill);
if(recipe.solidInput != null) this.decrStackSize(1, recipe.solidInput.stacksize);
tanks[2].setFill(tanks[2].getFill() + recipe.output);
}
public int getConsumption() {
return consumption;
}
protected DirPos[] getConPos() {
return new DirPos[] {
new DirPos(xCoord, yCoord - 1, zCoord, Library.NEG_Y),
new DirPos(xCoord + 1, yCoord, zCoord, Library.POS_X),
new DirPos(xCoord - 1, yCoord, zCoord, Library.NEG_X),
new DirPos(xCoord, yCoord, zCoord + 1, Library.POS_Z),
new DirPos(xCoord, yCoord, zCoord - 1, Library.NEG_Z),
};
}
@Override
public int[] getAccessibleSlotsFromSide(int meta) {
return new int[] { 1 };
}
@Override
public boolean isItemValidForSlot(int i, ItemStack itemStack) {
MixerRecipe[] recipes = MixerRecipes.getOutput(tanks[2].getTankType());
if(recipes == null || recipes.length <= 0) return false;
MixerRecipe recipe = recipes[this.recipeIndex % recipes.length];
if(recipe == null || recipe.solidInput == null) return false;
return recipe.solidInput.matchesRecipe(itemStack, true);
}
@Override
public void readFromNBT(NBTTagCompound nbt) {
super.readFromNBT(nbt);
this.power = nbt.getLong("power");
this.progress = nbt.getInteger("progress");
this.processTime = nbt.getInteger("processTime");
this.recipeIndex = nbt.getInteger("recipe");
for(int i = 0; i < 3; i++) this.tanks[i].readFromNBT(nbt, i + "");
}
@Override
public void writeToNBT(NBTTagCompound nbt) {
super.writeToNBT(nbt);
nbt.setLong("power", power);
nbt.setInteger("progress", progress);
nbt.setInteger("processTime", processTime);
nbt.setInteger("recipe", recipeIndex);
for(int i = 0; i < 3; i++) this.tanks[i].writeToNBT(nbt, i + "");
}
@Override
public long getPower() {
return power;
}
@Override
public void setPower(long power) {
this.power = power;
}
@Override
public long getMaxPower() {
return maxPower;
}
@Override
public FluidTank[] getAllTanks() {
return tanks;
}
@Override
public FluidTank[] getSendingTanks() {
return new FluidTank[] {tanks[2]};
}
@Override
public FluidTank[] getReceivingTanks() {
return new FluidTank[] {tanks[0], tanks[1]};
}
@Override
public Container provideContainer(int ID, EntityPlayer player, World world, int x, int y, int z) {
return new ContainerMixer(player.inventory, this);
}
@Override
@SideOnly(Side.CLIENT)
public Object provideGUI(int ID, EntityPlayer player, World world, int x, int y, int z) {
return new GUIMixer(player.inventory, this);
}
AxisAlignedBB aabb;
@Override
public AxisAlignedBB getRenderBoundingBox() {
if(aabb != null)
return aabb;
aabb = AxisAlignedBB.getBoundingBox(xCoord, yCoord, zCoord, xCoord + 1, yCoord + 3, zCoord + 1);
return aabb;
}
@Override
@SideOnly(Side.CLIENT)
public double getMaxRenderDistanceSquared() {
return 65536.0D;
}
@Override
public boolean hasPermission(EntityPlayer player) {
return player.getDistance(xCoord + 0.5, yCoord + 0.5, zCoord + 0.5) <= 16;
}
@Override
public void receiveControl(NBTTagCompound data) {
if(data.hasKey("toggle")) this.recipeIndex++;
}
@Override
public boolean canProvideInfo(UpgradeType type, int level, boolean extendedInfo) {
return type == UpgradeType.SPEED || type == UpgradeType.POWER || type == UpgradeType.OVERDRIVE;
}
@Override
public void provideInfo(UpgradeType type, int level, List<String> info, boolean extendedInfo) {
info.add(IUpgradeInfoProvider.getStandardLabel(ModBlocks.machine_mixer));
if(type == UpgradeType.SPEED) {
info.add(EnumChatFormatting.GREEN + I18nUtil.resolveKey(this.KEY_DELAY, "-" + (level * 25) + "%"));
info.add(EnumChatFormatting.RED + I18nUtil.resolveKey(this.KEY_CONSUMPTION, "+" + (level * 300) + "%"));
}
if(type == UpgradeType.POWER) {
info.add(EnumChatFormatting.GREEN + I18nUtil.resolveKey(this.KEY_CONSUMPTION, "-" + (level * 25) + "%"));
}
if(type == UpgradeType.OVERDRIVE) {
info.add((BobMathUtil.getBlink() ? EnumChatFormatting.RED : EnumChatFormatting.DARK_GRAY) + "YES");
}
}
@Override
public HashMap<UpgradeType, Integer> getValidUpgrades() {
HashMap<UpgradeType, Integer> upgrades = new HashMap<>();
upgrades.put(UpgradeType.SPEED, 3);
upgrades.put(UpgradeType.POWER, 3);
upgrades.put(UpgradeType.OVERDRIVE, 6);
return upgrades;
}
@Override
public FluidTank getTankToPaste() {
return this.tanks[2];
}
}