581 lines
16 KiB
Java
581 lines
16 KiB
Java
/*
|
|
* This file is part of Applied Energistics 2.
|
|
* Copyright (c) 2013 - 2015, 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 <http://www.gnu.org/licenses/lgpl>.
|
|
*/
|
|
|
|
package appeng.parts.misc;
|
|
|
|
|
|
import java.util.Arrays;
|
|
import java.util.Collections;
|
|
import java.util.List;
|
|
|
|
import net.minecraft.client.renderer.RenderBlocks;
|
|
import net.minecraft.entity.player.EntityPlayer;
|
|
import net.minecraft.inventory.IInventory;
|
|
import net.minecraft.item.ItemStack;
|
|
import net.minecraft.nbt.NBTTagCompound;
|
|
import net.minecraft.tileentity.TileEntity;
|
|
import net.minecraft.util.Vec3;
|
|
import net.minecraftforge.common.util.ForgeDirection;
|
|
|
|
import cpw.mods.fml.relauncher.Side;
|
|
import cpw.mods.fml.relauncher.SideOnly;
|
|
|
|
import buildcraft.api.transport.IPipeConnection;
|
|
import buildcraft.api.transport.IPipeTile.PipeType;
|
|
|
|
import appeng.api.AEApi;
|
|
import appeng.api.config.AccessRestriction;
|
|
import appeng.api.config.FuzzyMode;
|
|
import appeng.api.config.IncludeExclude;
|
|
import appeng.api.config.Settings;
|
|
import appeng.api.config.StorageFilter;
|
|
import appeng.api.config.Upgrades;
|
|
import appeng.api.networking.IGridNode;
|
|
import appeng.api.networking.events.MENetworkCellArrayUpdate;
|
|
import appeng.api.networking.events.MENetworkChannelsChanged;
|
|
import appeng.api.networking.events.MENetworkEventSubscribe;
|
|
import appeng.api.networking.events.MENetworkPowerStatusChange;
|
|
import appeng.api.networking.security.BaseActionSource;
|
|
import appeng.api.networking.security.MachineSource;
|
|
import appeng.api.networking.storage.IBaseMonitor;
|
|
import appeng.api.networking.ticking.IGridTickable;
|
|
import appeng.api.networking.ticking.ITickManager;
|
|
import appeng.api.networking.ticking.TickRateModulation;
|
|
import appeng.api.networking.ticking.TickingRequest;
|
|
import appeng.api.parts.IPartCollisionHelper;
|
|
import appeng.api.parts.IPartHost;
|
|
import appeng.api.parts.IPartRenderHelper;
|
|
import appeng.api.storage.ICellContainer;
|
|
import appeng.api.storage.IExternalStorageHandler;
|
|
import appeng.api.storage.IMEInventory;
|
|
import appeng.api.storage.IMEInventoryHandler;
|
|
import appeng.api.storage.IMEMonitor;
|
|
import appeng.api.storage.IMEMonitorHandlerReceiver;
|
|
import appeng.api.storage.StorageChannel;
|
|
import appeng.api.storage.data.IAEItemStack;
|
|
import appeng.api.storage.data.IItemList;
|
|
import appeng.api.util.IConfigManager;
|
|
import appeng.client.texture.CableBusTextures;
|
|
import appeng.core.settings.TickRates;
|
|
import appeng.core.stats.Achievements;
|
|
import appeng.core.sync.GuiBridge;
|
|
import appeng.helpers.IInterfaceHost;
|
|
import appeng.helpers.IPriorityHost;
|
|
import appeng.helpers.Reflected;
|
|
import appeng.integration.IntegrationType;
|
|
import appeng.me.GridAccessException;
|
|
import appeng.me.storage.MEInventoryHandler;
|
|
import appeng.me.storage.MEMonitorIInventory;
|
|
import appeng.parts.automation.PartUpgradeable;
|
|
import appeng.tile.inventory.AppEngInternalAEInventory;
|
|
import appeng.tile.inventory.InvOperation;
|
|
import appeng.transformer.annotations.Integration.Interface;
|
|
import appeng.transformer.annotations.Integration.Method;
|
|
import appeng.util.Platform;
|
|
import appeng.util.prioitylist.FuzzyPriorityList;
|
|
import appeng.util.prioitylist.PrecisePriorityList;
|
|
|
|
|
|
@Interface( iname = IntegrationType.BuildCraftTransport, iface = "buildcraft.api.transport.IPipeConnection" )
|
|
public class PartStorageBus extends PartUpgradeable implements IGridTickable, ICellContainer, IMEMonitorHandlerReceiver<IAEItemStack>, IPipeConnection, IPriorityHost
|
|
{
|
|
private final BaseActionSource mySrc;
|
|
private final AppEngInternalAEInventory Config = new AppEngInternalAEInventory( this, 63 );
|
|
private int priority = 0;
|
|
private boolean cached = false;
|
|
private MEMonitorIInventory monitor = null;
|
|
private MEInventoryHandler handler = null;
|
|
private int handlerHash = 0;
|
|
private boolean wasActive = false;
|
|
private byte resetCacheLogic = 0;
|
|
|
|
@Reflected
|
|
public PartStorageBus( final ItemStack is )
|
|
{
|
|
super( is );
|
|
this.getConfigManager().registerSetting( Settings.ACCESS, AccessRestriction.READ_WRITE );
|
|
this.getConfigManager().registerSetting( Settings.FUZZY_MODE, FuzzyMode.IGNORE_ALL );
|
|
this.getConfigManager().registerSetting( Settings.STORAGE_FILTER, StorageFilter.EXTRACTABLE_ONLY );
|
|
this.mySrc = new MachineSource( this );
|
|
}
|
|
|
|
@Override
|
|
@MENetworkEventSubscribe
|
|
public void powerRender( final MENetworkPowerStatusChange c )
|
|
{
|
|
this.updateStatus();
|
|
}
|
|
|
|
private void updateStatus()
|
|
{
|
|
final boolean currentActive = this.getProxy().isActive();
|
|
if( this.wasActive != currentActive )
|
|
{
|
|
this.wasActive = currentActive;
|
|
try
|
|
{
|
|
this.getProxy().getGrid().postEvent( new MENetworkCellArrayUpdate() );
|
|
this.getHost().markForUpdate();
|
|
}
|
|
catch( final GridAccessException e )
|
|
{
|
|
// :P
|
|
}
|
|
}
|
|
}
|
|
|
|
@MENetworkEventSubscribe
|
|
public void updateChannels( final MENetworkChannelsChanged changedChannels )
|
|
{
|
|
this.updateStatus();
|
|
}
|
|
|
|
@Override
|
|
protected int getUpgradeSlots()
|
|
{
|
|
return 5;
|
|
}
|
|
|
|
@Override
|
|
public void updateSetting( final IConfigManager manager, final Enum settingName, final Enum newValue )
|
|
{
|
|
this.resetCache( true );
|
|
this.getHost().markForSave();
|
|
}
|
|
|
|
@Override
|
|
public void onChangeInventory( final IInventory inv, final int slot, final InvOperation mc, final ItemStack removedStack, final ItemStack newStack )
|
|
{
|
|
super.onChangeInventory( inv, slot, mc, removedStack, newStack );
|
|
|
|
if( inv == this.Config )
|
|
{
|
|
this.resetCache( true );
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void upgradesChanged()
|
|
{
|
|
super.upgradesChanged();
|
|
this.resetCache( true );
|
|
}
|
|
|
|
@Override
|
|
public void readFromNBT( final NBTTagCompound data )
|
|
{
|
|
super.readFromNBT( data );
|
|
this.Config.readFromNBT( data, "config" );
|
|
this.priority = data.getInteger( "priority" );
|
|
}
|
|
|
|
@Override
|
|
public void writeToNBT( final NBTTagCompound data )
|
|
{
|
|
super.writeToNBT( data );
|
|
this.Config.writeToNBT( data, "config" );
|
|
data.setInteger( "priority", this.priority );
|
|
}
|
|
|
|
@Override
|
|
public IInventory getInventoryByName( final String name )
|
|
{
|
|
if( name.equals( "config" ) )
|
|
{
|
|
return this.Config;
|
|
}
|
|
|
|
return super.getInventoryByName( name );
|
|
}
|
|
|
|
private void resetCache( final boolean fullReset )
|
|
{
|
|
if( this.getHost() == null || this.getHost().getTile() == null || this.getHost().getTile().getWorldObj() == null || this.getHost().getTile().getWorldObj().isRemote )
|
|
{
|
|
return;
|
|
}
|
|
|
|
if( fullReset )
|
|
{
|
|
this.resetCacheLogic = 2;
|
|
}
|
|
else
|
|
{
|
|
this.resetCacheLogic = 1;
|
|
}
|
|
|
|
try
|
|
{
|
|
this.getProxy().getTick().alertDevice( this.getProxy().getNode() );
|
|
}
|
|
catch( final GridAccessException e )
|
|
{
|
|
// :P
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public boolean isValid( final Object verificationToken )
|
|
{
|
|
return this.handler == verificationToken;
|
|
}
|
|
|
|
@Override
|
|
public void postChange( final IBaseMonitor<IAEItemStack> monitor, final Iterable<IAEItemStack> change, final BaseActionSource source )
|
|
{
|
|
try
|
|
{
|
|
if( this.getProxy().isActive() )
|
|
{
|
|
this.getProxy().getStorage().postAlterationOfStoredItems( StorageChannel.ITEMS, change, this.mySrc );
|
|
}
|
|
}
|
|
catch( final GridAccessException e )
|
|
{
|
|
// :(
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onListUpdate()
|
|
{
|
|
// not used here.
|
|
}
|
|
|
|
@Override
|
|
public void getBoxes( final IPartCollisionHelper bch )
|
|
{
|
|
bch.addBox( 3, 3, 15, 13, 13, 16 );
|
|
bch.addBox( 2, 2, 14, 14, 14, 15 );
|
|
bch.addBox( 5, 5, 12, 11, 11, 14 );
|
|
}
|
|
|
|
@Override
|
|
@SideOnly( Side.CLIENT )
|
|
public void renderInventory( final IPartRenderHelper rh, final RenderBlocks renderer )
|
|
{
|
|
rh.setTexture( CableBusTextures.PartStorageSides.getIcon(), CableBusTextures.PartStorageSides.getIcon(), CableBusTextures.PartStorageBack.getIcon(), this.getItemStack().getIconIndex(), CableBusTextures.PartStorageSides.getIcon(), CableBusTextures.PartStorageSides.getIcon() );
|
|
|
|
rh.setBounds( 3, 3, 15, 13, 13, 16 );
|
|
rh.renderInventoryBox( renderer );
|
|
|
|
rh.setBounds( 2, 2, 14, 14, 14, 15 );
|
|
rh.renderInventoryBox( renderer );
|
|
|
|
rh.setBounds( 5, 5, 12, 11, 11, 14 );
|
|
rh.renderInventoryBox( renderer );
|
|
}
|
|
|
|
@Override
|
|
@SideOnly( Side.CLIENT )
|
|
public void renderStatic( final int x, final int y, final int z, final IPartRenderHelper rh, final RenderBlocks renderer )
|
|
{
|
|
this.setRenderCache( rh.useSimplifiedRendering( x, y, z, this, this.getRenderCache() ) );
|
|
rh.setTexture( CableBusTextures.PartStorageSides.getIcon(), CableBusTextures.PartStorageSides.getIcon(), CableBusTextures.PartStorageBack.getIcon(), this.getItemStack().getIconIndex(), CableBusTextures.PartStorageSides.getIcon(), CableBusTextures.PartStorageSides.getIcon() );
|
|
|
|
rh.setBounds( 3, 3, 15, 13, 13, 16 );
|
|
rh.renderBlock( x, y, z, renderer );
|
|
|
|
rh.setBounds( 2, 2, 14, 14, 14, 15 );
|
|
rh.renderBlock( x, y, z, renderer );
|
|
|
|
rh.setTexture( CableBusTextures.PartStorageSides.getIcon(), CableBusTextures.PartStorageSides.getIcon(), CableBusTextures.PartStorageBack.getIcon(), this.getItemStack().getIconIndex(), CableBusTextures.PartStorageSides.getIcon(), CableBusTextures.PartStorageSides.getIcon() );
|
|
|
|
rh.setBounds( 5, 5, 12, 11, 11, 13 );
|
|
rh.renderBlock( x, y, z, renderer );
|
|
|
|
rh.setTexture( CableBusTextures.PartMonitorSidesStatus.getIcon(), CableBusTextures.PartMonitorSidesStatus.getIcon(), CableBusTextures.PartMonitorBack.getIcon(), this.getItemStack().getIconIndex(), CableBusTextures.PartMonitorSidesStatus.getIcon(), CableBusTextures.PartMonitorSidesStatus.getIcon() );
|
|
|
|
rh.setBounds( 5, 5, 13, 11, 11, 14 );
|
|
rh.renderBlock( x, y, z, renderer );
|
|
|
|
this.renderLights( x, y, z, rh, renderer );
|
|
}
|
|
|
|
@Override
|
|
public void onNeighborChanged()
|
|
{
|
|
this.resetCache( false );
|
|
}
|
|
|
|
@Override
|
|
public int cableConnectionRenderTo()
|
|
{
|
|
return 4;
|
|
}
|
|
|
|
@Override
|
|
public boolean onPartActivate( final EntityPlayer player, final Vec3 pos )
|
|
{
|
|
if( !player.isSneaking() )
|
|
{
|
|
if( Platform.isClient() )
|
|
{
|
|
return true;
|
|
}
|
|
|
|
Platform.openGUI( player, this.getHost().getTile(), this.getSide(), GuiBridge.GUI_STORAGEBUS );
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
public TickingRequest getTickingRequest( final IGridNode node )
|
|
{
|
|
return new TickingRequest( TickRates.StorageBus.getMin(), TickRates.StorageBus.getMax(), this.monitor == null, true );
|
|
}
|
|
|
|
@Override
|
|
public TickRateModulation tickingRequest( final IGridNode node, final int ticksSinceLastCall )
|
|
{
|
|
if( this.resetCacheLogic != 0 )
|
|
{
|
|
this.resetCache();
|
|
}
|
|
|
|
if( this.monitor != null )
|
|
{
|
|
return this.monitor.onTick();
|
|
}
|
|
|
|
return TickRateModulation.SLEEP;
|
|
}
|
|
|
|
private void resetCache()
|
|
{
|
|
final boolean fullReset = this.resetCacheLogic == 2;
|
|
this.resetCacheLogic = 0;
|
|
|
|
final IMEInventory<IAEItemStack> in = this.getInternalHandler();
|
|
IItemList<IAEItemStack> before = AEApi.instance().storage().createItemList();
|
|
if( in != null )
|
|
{
|
|
before = in.getAvailableItems( before );
|
|
}
|
|
|
|
this.cached = false;
|
|
if( fullReset )
|
|
{
|
|
this.handlerHash = 0;
|
|
}
|
|
|
|
final IMEInventory<IAEItemStack> out = this.getInternalHandler();
|
|
|
|
if( this.monitor != null )
|
|
{
|
|
this.monitor.onTick();
|
|
}
|
|
|
|
IItemList<IAEItemStack> after = AEApi.instance().storage().createItemList();
|
|
if( out != null )
|
|
{
|
|
after = out.getAvailableItems( after );
|
|
}
|
|
|
|
Platform.postListChanges( before, after, this, this.mySrc );
|
|
}
|
|
|
|
public MEInventoryHandler getInternalHandler()
|
|
{
|
|
if( this.cached )
|
|
{
|
|
return this.handler;
|
|
}
|
|
|
|
final boolean wasSleeping = this.monitor == null;
|
|
|
|
this.cached = true;
|
|
final TileEntity self = this.getHost().getTile();
|
|
final TileEntity target = self.getWorldObj().getTileEntity( self.xCoord + this.getSide().offsetX, self.yCoord + this.getSide().offsetY, self.zCoord + this.getSide().offsetZ );
|
|
|
|
final int newHandlerHash = Platform.generateTileHash( target );
|
|
|
|
if( this.handlerHash == newHandlerHash && this.handlerHash != 0 )
|
|
{
|
|
return this.handler;
|
|
}
|
|
|
|
try
|
|
{
|
|
// force grid to update handlers...
|
|
this.getProxy().getGrid().postEvent( new MENetworkCellArrayUpdate() );
|
|
}
|
|
catch( final GridAccessException e )
|
|
{
|
|
// :3
|
|
}
|
|
|
|
this.handlerHash = newHandlerHash;
|
|
this.handler = null;
|
|
this.monitor = null;
|
|
if( target != null )
|
|
{
|
|
final IExternalStorageHandler esh = AEApi.instance().registries().externalStorage().getHandler( target, this.getSide().getOpposite(), StorageChannel.ITEMS, this.mySrc );
|
|
if( esh != null )
|
|
{
|
|
final IMEInventory inv = esh.getInventory( target, this.getSide().getOpposite(), StorageChannel.ITEMS, this.mySrc );
|
|
|
|
if( inv instanceof MEMonitorIInventory )
|
|
{
|
|
final MEMonitorIInventory h = (MEMonitorIInventory) inv;
|
|
h.setMode( (StorageFilter) this.getConfigManager().getSetting( Settings.STORAGE_FILTER ) );
|
|
h.setActionSource( new MachineSource( this ) );
|
|
}
|
|
|
|
if( inv instanceof MEMonitorIInventory )
|
|
{
|
|
this.monitor = (MEMonitorIInventory) inv;
|
|
}
|
|
|
|
if( inv != null )
|
|
{
|
|
this.checkInterfaceVsStorageBus( target, this.getSide().getOpposite() );
|
|
|
|
this.handler = new MEInventoryHandler( inv, StorageChannel.ITEMS );
|
|
|
|
this.handler.setBaseAccess( (AccessRestriction) this.getConfigManager().getSetting( Settings.ACCESS ) );
|
|
this.handler.setWhitelist( this.getInstalledUpgrades( Upgrades.INVERTER ) > 0 ? IncludeExclude.BLACKLIST : IncludeExclude.WHITELIST );
|
|
this.handler.setPriority( this.priority );
|
|
|
|
final IItemList<IAEItemStack> priorityList = AEApi.instance().storage().createItemList();
|
|
|
|
final int slotsToUse = 18 + this.getInstalledUpgrades( Upgrades.CAPACITY ) * 9;
|
|
for( int x = 0; x < this.Config.getSizeInventory() && x < slotsToUse; x++ )
|
|
{
|
|
final IAEItemStack is = this.Config.getAEStackInSlot( x );
|
|
if( is != null )
|
|
{
|
|
priorityList.add( is );
|
|
}
|
|
}
|
|
|
|
if( this.getInstalledUpgrades( Upgrades.FUZZY ) > 0 )
|
|
{
|
|
this.handler.setPartitionList( new FuzzyPriorityList( priorityList, (FuzzyMode) this.getConfigManager().getSetting( Settings.FUZZY_MODE ) ) );
|
|
}
|
|
else
|
|
{
|
|
this.handler.setPartitionList( new PrecisePriorityList( priorityList ) );
|
|
}
|
|
|
|
if( inv instanceof IMEMonitor )
|
|
{
|
|
( (IBaseMonitor) inv ).addListener( this, this.handler );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// update sleep state...
|
|
if( wasSleeping != ( this.monitor == null ) )
|
|
{
|
|
try
|
|
{
|
|
final ITickManager tm = this.getProxy().getTick();
|
|
if( this.monitor == null )
|
|
{
|
|
tm.sleepDevice( this.getProxy().getNode() );
|
|
}
|
|
else
|
|
{
|
|
tm.wakeDevice( this.getProxy().getNode() );
|
|
}
|
|
}
|
|
catch( final GridAccessException e )
|
|
{
|
|
// :(
|
|
}
|
|
}
|
|
|
|
return this.handler;
|
|
}
|
|
|
|
private void checkInterfaceVsStorageBus( final TileEntity target, final ForgeDirection side )
|
|
{
|
|
IInterfaceHost achievement = null;
|
|
|
|
if( target instanceof IInterfaceHost )
|
|
{
|
|
achievement = (IInterfaceHost) target;
|
|
}
|
|
|
|
if( target instanceof IPartHost )
|
|
{
|
|
final Object part = ( (IPartHost) target ).getPart( side );
|
|
if( part instanceof IInterfaceHost )
|
|
{
|
|
achievement = (IInterfaceHost) part;
|
|
}
|
|
}
|
|
|
|
if( achievement != null && achievement.getActionableNode() != null )
|
|
{
|
|
Platform.addStat( achievement.getActionableNode().getPlayerID(), Achievements.Recursive.getAchievement() );
|
|
// Platform.addStat( getActionableNode().getPlayerID(), Achievements.Recursive.getAchievement() );
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public List<IMEInventoryHandler> getCellArray( final StorageChannel channel )
|
|
{
|
|
if( channel == StorageChannel.ITEMS )
|
|
{
|
|
final IMEInventoryHandler out = this.getProxy().isActive() ? this.getInternalHandler() : null;
|
|
if( out != null )
|
|
{
|
|
return Collections.singletonList( out );
|
|
}
|
|
}
|
|
return Arrays.asList( new IMEInventoryHandler[] {} );
|
|
}
|
|
|
|
@Override
|
|
public int getPriority()
|
|
{
|
|
return this.priority;
|
|
}
|
|
|
|
@Override
|
|
public void setPriority( final int newValue )
|
|
{
|
|
this.priority = newValue;
|
|
this.getHost().markForSave();
|
|
this.resetCache( true );
|
|
}
|
|
|
|
@Override
|
|
public void blinkCell( final int slot )
|
|
{
|
|
}
|
|
|
|
@Override
|
|
@Method( iname = IntegrationType.BuildCraftTransport )
|
|
public ConnectOverride overridePipeConnection( final PipeType type, final ForgeDirection with )
|
|
{
|
|
return type == PipeType.ITEM && with == this.getSide() ? ConnectOverride.CONNECT : ConnectOverride.DISCONNECT;
|
|
}
|
|
|
|
@Override
|
|
public void saveChanges( final IMEInventory cellInventory )
|
|
{
|
|
// nope!
|
|
}
|
|
}
|