package de.ellpeck.actuallyadditions.mod.tile; import java.util.ArrayList; import java.util.Collections; import java.util.HashMap; import java.util.List; import java.util.Map; import org.cyclops.commoncapabilities.api.capability.itemhandler.ISlotlessItemHandler; import org.cyclops.commoncapabilities.capability.itemhandler.SlotlessItemHandlerConfig; import de.ellpeck.actuallyadditions.api.laser.Network; import de.ellpeck.actuallyadditions.mod.ActuallyAdditions; import de.ellpeck.actuallyadditions.mod.network.PacketHandler; import de.ellpeck.actuallyadditions.mod.network.PacketServerToClient; import de.ellpeck.actuallyadditions.mod.util.StackUtil; import de.ellpeck.actuallyadditions.mod.util.WorldUtil; import de.ellpeck.actuallyadditions.mod.util.compat.CommonCapsUtil; import de.ellpeck.actuallyadditions.mod.util.compat.SlotlessableItemHandlerWrapper; import net.minecraft.entity.player.EntityPlayer; import net.minecraft.entity.player.EntityPlayerMP; import net.minecraft.item.ItemStack; import net.minecraft.nbt.NBTTagCompound; import net.minecraft.tileentity.TileEntity; import net.minecraft.util.EnumFacing; import net.minecraft.util.math.BlockPos; import net.minecraftforge.common.capabilities.Capability; import net.minecraftforge.items.IItemHandler; public class TileEntityItemViewer extends TileEntityBase { public final List genericInfos = new ArrayList<>(); public final Map itemHandlerInfos = new HashMap<>(); public final List slotlessInfos = new ArrayList<>(); protected final SlotlessableItemHandlerWrapper itemHandler; public TileEntityLaserRelayItem connectedRelay; private int lastNetworkChangeAmount = -1; public TileEntityItemViewer(String name) { super(name); IItemHandler normalHandler = new IItemHandler() { @Override public int getSlots() { int size = 0; List infos = TileEntityItemViewer.this.getItemHandlerInfos(); if (infos != null) { for (GenericItemHandlerInfo info : infos) { if (info.isLoaded()) { for (SlotlessableItemHandlerWrapper handler : info.handlers) { IItemHandler normalHandler = handler.getNormalHandler(); if (normalHandler != null) { size += normalHandler.getSlots(); } } } } } return size; } @Override public ItemStack getStackInSlot(int slot) { IItemHandlerInfo handler = TileEntityItemViewer.this.getSwitchedIndexHandler(slot); if (handler != null && handler.isLoaded()) { return handler.handler.getStackInSlot(handler.switchedIndex); } return StackUtil.getEmpty(); } @Override public ItemStack insertItem(int slot, ItemStack stack, boolean simulate) { IItemHandlerInfo info = TileEntityItemViewer.this.getSwitchedIndexHandler(slot); if (info != null && info.isLoaded() && TileEntityItemViewer.this.isWhitelisted(info, stack, false)) { ItemStack remain = info.handler.insertItem(info.switchedIndex, stack, simulate); if (!ItemStack.areItemStacksEqual(remain, stack) && !simulate) { TileEntityItemViewer.this.markDirty(); TileEntityItemViewer.this.doItemParticle(stack, info.relayInQuestion.getPos(), TileEntityItemViewer.this.connectedRelay.getPos()); } return remain; } return stack; } @Override public ItemStack extractItem(int slot, int amount, boolean simulate) { ItemStack stackIn = this.getStackInSlot(slot); if (StackUtil.isValid(stackIn)) { IItemHandlerInfo info = TileEntityItemViewer.this.getSwitchedIndexHandler(slot); if (info != null && info.isLoaded() && TileEntityItemViewer.this.isWhitelisted(info, stackIn, true)) { ItemStack extracted = info.handler.extractItem(info.switchedIndex, amount, simulate); if (StackUtil.isValid(extracted) && !simulate) { TileEntityItemViewer.this.markDirty(); TileEntityItemViewer.this.doItemParticle(extracted, TileEntityItemViewer.this.connectedRelay.getPos(), info.relayInQuestion.getPos()); } return extracted; } } return StackUtil.getEmpty(); } @Override public int getSlotLimit(int slot) { IItemHandlerInfo info = TileEntityItemViewer.this.getSwitchedIndexHandler(slot); if (info != null && info.isLoaded()) { return info.handler.getSlotLimit(info.switchedIndex); } else { return 0; } } }; Object slotlessHandler = null; if (ActuallyAdditions.commonCapsLoaded) { slotlessHandler = CommonCapsUtil.createSlotlessItemViewerHandler(this, normalHandler); } this.itemHandler = new SlotlessableItemHandlerWrapper(normalHandler, slotlessHandler); } public TileEntityItemViewer() { this("itemViewer"); } @Override public IItemHandler getItemHandler(EnumFacing facing) { return this.itemHandler.getNormalHandler(); } @SuppressWarnings("unchecked") @Override public T getCapability(Capability capability, EnumFacing facing) { if (ActuallyAdditions.commonCapsLoaded) { if (capability == SlotlessItemHandlerConfig.CAPABILITY) { Object handler = this.itemHandler.getSlotlessHandler(); if (handler != null) { return (T) handler; } } } return super.getCapability(capability, facing); } private List getItemHandlerInfos() { this.queryAndSaveData(); return this.genericInfos; } public void doItemParticle(ItemStack stack, BlockPos input, BlockPos output) { if (!this.world.isRemote) { NBTTagCompound compound = new NBTTagCompound(); stack.writeToNBT(compound); compound.setDouble("InX", input.getX()); compound.setDouble("InY", input.getY()); compound.setDouble("InZ", input.getZ()); compound.setDouble("OutX", output.getX()); compound.setDouble("OutY", output.getY()); compound.setDouble("OutZ", output.getZ()); int rangeSq = 16 * 16; for (EntityPlayer player : this.world.playerEntities) { if (player instanceof EntityPlayerMP) { if (player.getDistanceSq(input) <= rangeSq || player.getDistanceSq(output) <= rangeSq) { PacketHandler.theNetwork.sendTo(new PacketServerToClient(compound, PacketHandler.LASER_PARTICLE_HANDLER), (EntityPlayerMP) player); } } } } } private void queryAndSaveData() { if (this.connectedRelay != null) { Network network = this.connectedRelay.getNetwork(); if (network != null) { if (this.lastNetworkChangeAmount != network.changeAmount) { this.clearInfos(); this.connectedRelay.getItemHandlersInNetwork(network, this.genericInfos); if (!this.genericInfos.isEmpty()) { Collections.sort(this.genericInfos); int slotsQueried = 0; for (GenericItemHandlerInfo info : this.genericInfos) { for (SlotlessableItemHandlerWrapper handler : info.handlers) { IItemHandler normalHandler = handler.getNormalHandler(); if (normalHandler != null) { for (int i = 0; i < normalHandler.getSlots(); i++) { this.itemHandlerInfos.put(slotsQueried, new IItemHandlerInfo(normalHandler, i, info.relayInQuestion)); slotsQueried++; } } if (ActuallyAdditions.commonCapsLoaded) { Object slotlessHandler = handler.getSlotlessHandler(); if (slotlessHandler instanceof ISlotlessItemHandler) { this.slotlessInfos.add(new SlotlessItemHandlerInfo(slotlessHandler, info.relayInQuestion)); } } } } } this.lastNetworkChangeAmount = network.changeAmount; } return; } } this.clearInfos(); this.lastNetworkChangeAmount = -1; } private void clearInfos() { if (!this.genericInfos.isEmpty()) { this.genericInfos.clear(); } if (!this.itemHandlerInfos.isEmpty()) { this.itemHandlerInfos.clear(); } if (!this.slotlessInfos.isEmpty()) { this.slotlessInfos.clear(); } } private IItemHandlerInfo getSwitchedIndexHandler(int i) { this.queryAndSaveData(); return this.itemHandlerInfos.get(i); } @Override public boolean shouldSaveDataOnChangeOrWorldStart() { return true; } @Override public void saveDataOnChangeOrWorldStart() { TileEntityLaserRelayItem tileFound = null; if (this.world != null) { //Why is that even possible..? for (int i = 0; i <= 5; i++) { EnumFacing side = WorldUtil.getDirectionBySidesInOrder(i); BlockPos pos = this.getPos().offset(side); if (this.world.isBlockLoaded(pos)) { TileEntity tile = this.world.getTileEntity(pos); if (tile instanceof TileEntityLaserRelayItem) { if (tileFound != null) { this.connectedRelay = null; return; } else { tileFound = (TileEntityLaserRelayItem) tile; } } } } } this.connectedRelay = tileFound; } public boolean isWhitelisted(SpecificItemHandlerInfo handler, ItemStack stack, boolean output) { boolean whitelisted = handler.relayInQuestion.isWhitelisted(stack, output); TileEntityLaserRelayItem connected = this.connectedRelay; if (connected != null && connected != handler.relayInQuestion) { return whitelisted && connected.isWhitelisted(stack, output); } else { return whitelisted; } } public static class SlotlessItemHandlerInfo extends SpecificItemHandlerInfo { public final Object handler; public SlotlessItemHandlerInfo(Object handler, TileEntityLaserRelayItem relayInQuestion) { super(relayInQuestion); this.handler = handler; } } private static class IItemHandlerInfo extends SpecificItemHandlerInfo { public final IItemHandler handler; public final int switchedIndex; public IItemHandlerInfo(IItemHandler handler, int switchedIndex, TileEntityLaserRelayItem relayInQuestion) { super(relayInQuestion); this.handler = handler; this.switchedIndex = switchedIndex; } } private static class SpecificItemHandlerInfo { public final TileEntityLaserRelayItem relayInQuestion; public SpecificItemHandlerInfo(TileEntityLaserRelayItem relayInQuestion) { this.relayInQuestion = relayInQuestion; } public boolean isLoaded() { return this.relayInQuestion.hasWorld() && this.relayInQuestion.getWorld().isBlockLoaded(this.relayInQuestion.getPos()); } } public static class GenericItemHandlerInfo implements Comparable { public final List handlers = new ArrayList<>(); public final TileEntityLaserRelayItem relayInQuestion; public GenericItemHandlerInfo(TileEntityLaserRelayItem relayInQuestion) { this.relayInQuestion = relayInQuestion; } public boolean isLoaded() { return this.relayInQuestion.hasWorld() && this.relayInQuestion.getWorld().isBlockLoaded(this.relayInQuestion.getPos()); } @Override public int compareTo(GenericItemHandlerInfo other) { int thisPrio = this.relayInQuestion.getPriority(); int otherPrio = other.relayInQuestion.getPriority(); if (thisPrio == otherPrio) { return 0; } else if (thisPrio > otherPrio) { return -1; } else { return 1; } } } }