/* * This file is part of Applied Energistics 2. * Copyright (c) 2013 - 2014, AlgorithmX2, All rights reserved. * * Applied Energistics 2 is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * Applied Energistics 2 is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with Applied Energistics 2. If not, see . */ package appeng.tile.misc; import java.io.IOException; import java.util.EnumSet; import java.util.List; import javax.annotation.Nullable; import appeng.api.AEApi; import appeng.api.config.Actionable; import appeng.api.config.PowerMultiplier; import appeng.api.config.Upgrades; import appeng.api.definitions.IComparableDefinition; import appeng.api.definitions.ITileDefinition; import appeng.api.features.IInscriberRecipe; import appeng.api.features.InscriberProcessType; import appeng.api.implementations.IUpgradeableHost; import appeng.api.networking.IGridNode; import appeng.api.networking.energy.IEnergyGrid; import appeng.api.networking.energy.IEnergySource; import appeng.api.networking.ticking.IGridTickable; import appeng.api.networking.ticking.TickRateModulation; import appeng.api.networking.ticking.TickingRequest; import appeng.api.util.AECableType; import appeng.api.util.IConfigManager; import appeng.core.features.registries.entries.InscriberRecipe; import appeng.core.settings.TickRates; import appeng.helpers.Reflected; import appeng.me.GridAccessException; import appeng.parts.automation.DefinitionUpgradeInventory; import appeng.parts.automation.UpgradeInventory; import appeng.tile.TileEvent; import appeng.tile.events.TileEventType; import appeng.tile.grid.AENetworkPowerTile; import appeng.tile.inventory.AppEngInternalInventory; import appeng.tile.inventory.InvOperation; import appeng.util.ConfigManager; import appeng.util.IConfigManagerHost; import appeng.util.InventoryAdaptor; import appeng.util.Platform; import appeng.util.inv.WrapperInventoryRange; import appeng.util.item.AEItemStack; import com.google.common.collect.Lists; import io.netty.buffer.ByteBuf; import net.minecraft.inventory.IInventory; import net.minecraft.item.ItemStack; import net.minecraft.nbt.NBTTagCompound; import net.minecraft.world.World; import net.minecraftforge.common.util.ForgeDirection; /** * @author AlgorithmX2 * @author thatsIch * @version rv2 * @since rv0 */ public class TileInscriber extends AENetworkPowerTile implements IGridTickable, IUpgradeableHost, IConfigManagerHost { private final int maxProcessingTime = 100; private final int[] top = { 0 }; private final int[] bottom = { 1 }; private final int[] sides = { 2, 3 }; private final AppEngInternalInventory inv = new AppEngInternalInventory(this, 4); private final IConfigManager settings; private final UpgradeInventory upgrades; private int processingTime = 0; // cycles from 0 - 16, at 8 it preforms the action, at 16 it re-enables the normal // routine. private boolean smash; private int finalStep; private long clientStart; @Reflected public TileInscriber() { this.getProxy().setValidSides(EnumSet.noneOf(ForgeDirection.class)); this.setInternalMaxPower(1500); this.getProxy().setIdlePowerUsage(0); this.settings = new ConfigManager(this); final ITileDefinition inscriberDefinition = AEApi.instance().definitions().blocks().inscriber(); this.upgrades = new DefinitionUpgradeInventory( inscriberDefinition, this, this.getUpgradeSlots() ); } private int getUpgradeSlots() { return 3; } @Override public AECableType getCableConnectionType(final ForgeDirection dir) { return AECableType.COVERED; } @TileEvent(TileEventType.WORLD_NBT_WRITE) public void writeToNBT_TileInscriber(final NBTTagCompound data) { this.inv.writeToNBT(data, "inscriberInv"); this.upgrades.writeToNBT(data, "upgrades"); this.settings.writeToNBT(data); } @TileEvent(TileEventType.WORLD_NBT_READ) public void readFromNBT_TileInscriber(final NBTTagCompound data) { this.inv.readFromNBT(data, "inscriberInv"); this.upgrades.readFromNBT(data, "upgrades"); this.settings.readFromNBT(data); } @TileEvent(TileEventType.NETWORK_READ) public boolean readFromStream_TileInscriber(final ByteBuf data) throws IOException { final int slot = data.readByte(); final boolean oldSmash = this.isSmash(); final boolean newSmash = (slot & 64) == 64; if (oldSmash != newSmash && newSmash) { this.setSmash(true); this.setClientStart(System.currentTimeMillis()); } for (int num = 0; num < this.inv.getSizeInventory(); num++) { if ((slot & (1 << num)) > 0) { this.inv.setInventorySlotContents( num, AEItemStack.loadItemStackFromPacket(data).getItemStack() ); } else { this.inv.setInventorySlotContents(num, null); } } return false; } @TileEvent(TileEventType.NETWORK_WRITE) public void writeToStream_TileInscriber(final ByteBuf data) throws IOException { int slot = this.isSmash() ? 64 : 0; for (int num = 0; num < this.inv.getSizeInventory(); num++) { if (this.inv.getStackInSlot(num) != null) { slot |= (1 << num); } } data.writeByte(slot); for (int num = 0; num < this.inv.getSizeInventory(); num++) { if ((slot & (1 << num)) > 0) { final AEItemStack st = AEItemStack.create(this.inv.getStackInSlot(num)); st.writeToPacket(data); } } } @Override public void setOrientation(final ForgeDirection inForward, final ForgeDirection inUp) { super.setOrientation(inForward, inUp); this.getProxy().setValidSides(EnumSet.complementOf(EnumSet.of(this.getForward())) ); this.setPowerSides(EnumSet.complementOf(EnumSet.of(this.getForward()))); } @Override public void getDrops( final World w, final int x, final int y, final int z, final List drops ) { super.getDrops(w, x, y, z, drops); for (int h = 0; h < this.upgrades.getSizeInventory(); h++) { final ItemStack is = this.upgrades.getStackInSlot(h); if (is != null) { drops.add(is); } } } @Override public boolean requiresTESR() { return true; } @Override public IInventory getInternalInventory() { return this.inv; } @Override public int getInventoryStackLimit() { return 1; } @Override public boolean isItemValidForSlot(final int i, final ItemStack itemstack) { if (this.isSmash()) { return false; } if (i == 0 || i == 1) { if (AEApi.instance().definitions().materials().namePress().isSameAs(itemstack )) { return true; } for (final ItemStack optionals : AEApi.instance().registries().inscriber().getOptionals()) { if (Platform.isSameItemPrecise(optionals, itemstack)) { return true; } } } return i == 2; } @Override public void onChangeInventory( final IInventory inv, final int slot, final InvOperation mc, final ItemStack removed, final ItemStack added ) { try { if (mc != InvOperation.markDirty) { if (slot != 3) { this.setProcessingTime(0); } if (!this.isSmash()) { this.markForUpdate(); } this.getProxy().getTick().wakeDevice(this.getProxy().getNode()); } } catch (final GridAccessException e) { // :P } } @Override public boolean canExtractItem(final int slotIndex, final ItemStack extractedItem, final int side) { if (this.isSmash()) { return false; } return slotIndex == 0 || slotIndex == 1 || slotIndex == 3; } @Override public int[] getAccessibleSlotsBySide(final ForgeDirection d) { if (d == ForgeDirection.UP) { return this.top; } if (d == ForgeDirection.DOWN) { return this.bottom; } return this.sides; } @Override public TickingRequest getTickingRequest(final IGridNode node) { return new TickingRequest( TickRates.Inscriber.getMin(), TickRates.Inscriber.getMax(), !this.hasWork(), false ); } private boolean hasWork() { if (this.getTask() != null) { return true; } this.setProcessingTime(0); return this.isSmash(); } @Nullable public IInscriberRecipe getTask() { final ItemStack plateA = this.getStackInSlot(0); final ItemStack plateB = this.getStackInSlot(1); ItemStack renamedItem = this.getStackInSlot(2); if (plateA != null && plateA.stackSize > 1) { return null; } if (plateB != null && plateB.stackSize > 1) { return null; } if (renamedItem != null && renamedItem.stackSize > 1) { return null; } final IComparableDefinition namePress = AEApi.instance().definitions().materials().namePress(); final boolean isNameA = namePress.isSameAs(plateA); final boolean isNameB = namePress.isSameAs(plateB); if ((isNameA || isNameB) && (isNameA || plateA == null) && (isNameB || plateB == null)) { if (renamedItem != null) { String name = ""; if (plateA != null) { final NBTTagCompound tag = Platform.openNbtData(plateA); name += tag.getString("InscribeName"); } if (plateB != null) { final NBTTagCompound tag = Platform.openNbtData(plateB); if (name.length() > 0) { name += " "; } name += tag.getString("InscribeName"); } final ItemStack startingItem = renamedItem.copy(); renamedItem = renamedItem.copy(); final NBTTagCompound tag = Platform.openNbtData(renamedItem); final NBTTagCompound display = tag.getCompoundTag("display"); tag.setTag("display", display); if (name.length() > 0) { display.setString("Name", name); } else { display.removeTag("Name"); } final List inputs = Lists.newArrayList(startingItem); final InscriberProcessType type = InscriberProcessType.Inscribe; return new InscriberRecipe(inputs, renamedItem, plateA, plateB, type); } } for (final IInscriberRecipe recipe : AEApi.instance().registries().inscriber().getRecipes()) { final boolean matchA = (plateA == null && !recipe.getTopOptional().isPresent()) || (Platform.isSameItemPrecise(plateA, recipe.getTopOptional().orNull())) && // and... (plateB == null && !recipe.getBottomOptional().isPresent()) | (Platform.isSameItemPrecise( plateB, recipe.getBottomOptional().orNull() )); final boolean matchB = (plateB == null && !recipe.getTopOptional().isPresent()) || (Platform.isSameItemPrecise(plateB, recipe.getTopOptional().orNull())) && // and... (plateA == null && !recipe.getBottomOptional().isPresent()) | (Platform.isSameItemPrecise( plateA, recipe.getBottomOptional().orNull() )); if (matchA || matchB) { for (final ItemStack option : recipe.getInputs()) { if (Platform.isSameItemPrecise(option, this.getStackInSlot(2))) { return recipe; } } } } return null; } @Override public TickRateModulation tickingRequest(final IGridNode node, final int ticksSinceLastCall) { if (this.isSmash()) { this.finalStep++; if (this.finalStep == 8) { final IInscriberRecipe out = this.getTask(); if (out != null) { final ItemStack outputCopy = out.getOutput().copy(); final InventoryAdaptor ad = InventoryAdaptor.getAdaptor( new WrapperInventoryRange(this.inv, 3, 1, true), ForgeDirection.UNKNOWN ); if (ad.addItems(outputCopy) == null) { this.setProcessingTime(0); if (out.getProcessType() == InscriberProcessType.Press) { this.setInventorySlotContents(0, null); this.setInventorySlotContents(1, null); } this.setInventorySlotContents(2, null); } } this.markDirty(); } else if (this.finalStep == 16) { this.finalStep = 0; this.setSmash(false); this.markForUpdate(); } } else { try { final IEnergyGrid eg = this.getProxy().getEnergy(); IEnergySource src = this; // Base 1, increase by 1 for each card final int speedFactor = 1 + this.upgrades.getInstalledUpgrades(Upgrades.SPEED); final int powerConsumption = 10 * speedFactor; final double powerThreshold = powerConsumption - 0.01; double powerReq = this.extractAEPower( powerConsumption, Actionable.SIMULATE, PowerMultiplier.CONFIG ); if (powerReq <= powerThreshold) { src = eg; powerReq = eg.extractAEPower( powerConsumption, Actionable.SIMULATE, PowerMultiplier.CONFIG ); } if (powerReq > powerThreshold) { src.extractAEPower( powerConsumption, Actionable.MODULATE, PowerMultiplier.CONFIG ); if (this.getProcessingTime() == 0) { this.setProcessingTime(this.getProcessingTime() + speedFactor); } else { this.setProcessingTime( this.getProcessingTime() + ticksSinceLastCall * speedFactor ); } } } catch (final GridAccessException e) { // :P } if (this.getProcessingTime() > this.getMaxProcessingTime()) { this.setProcessingTime(this.getMaxProcessingTime()); final IInscriberRecipe out = this.getTask(); if (out != null) { final ItemStack outputCopy = out.getOutput().copy(); final InventoryAdaptor ad = InventoryAdaptor.getAdaptor( new WrapperInventoryRange(this.inv, 3, 1, true), ForgeDirection.UNKNOWN ); if (ad.simulateAdd(outputCopy) == null) { this.setSmash(true); this.finalStep = 0; this.markForUpdate(); } } } } return this.hasWork() ? TickRateModulation.URGENT : TickRateModulation.SLEEP; } @Override public IConfigManager getConfigManager() { return this.settings; } @Override public IInventory getInventoryByName(final String name) { if (name.equals("inv")) { return this.inv; } if (name.equals("upgrades")) { return this.upgrades; } return null; } @Override public int getInstalledUpgrades(final Upgrades u) { return this.upgrades.getInstalledUpgrades(u); } @Override public void updateSetting( final IConfigManager manager, final Enum settingName, final Enum newValue ) {} public long getClientStart() { return this.clientStart; } private void setClientStart(final long clientStart) { this.clientStart = clientStart; } public boolean isSmash() { return this.smash; } public void setSmash(final boolean smash) { this.smash = smash; } public int getMaxProcessingTime() { return this.maxProcessingTime; } public int getProcessingTime() { return this.processingTime; } private void setProcessingTime(final int processingTime) { this.processingTime = processingTime; } }