Package appeng.parts.automation

Source Code of appeng.parts.automation.PartAnnihilationPlane

package appeng.parts.automation;

import java.util.LinkedList;
import java.util.List;
import java.util.concurrent.Callable;

import net.minecraft.block.Block;
import net.minecraft.block.material.Material;
import net.minecraft.client.renderer.RenderBlocks;
import net.minecraft.entity.Entity;
import net.minecraft.entity.item.EntityItem;
import net.minecraft.init.Blocks;
import net.minecraft.item.ItemStack;
import net.minecraft.nbt.NBTTagCompound;
import net.minecraft.tileentity.TileEntity;
import net.minecraft.util.AxisAlignedBB;
import net.minecraft.world.WorldServer;
import net.minecraftforge.common.util.ForgeDirection;
import appeng.api.config.Actionable;
import appeng.api.config.PowerMultiplier;
import appeng.api.networking.IGridNode;
import appeng.api.networking.energy.IEnergyGrid;
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.IStorageGrid;
import appeng.api.networking.ticking.IGridTickable;
import appeng.api.networking.ticking.TickRateModulation;
import appeng.api.networking.ticking.TickingRequest;
import appeng.api.parts.IPart;
import appeng.api.parts.IPartCollisionHelper;
import appeng.api.parts.IPartHost;
import appeng.api.parts.IPartRenderHelper;
import appeng.api.storage.data.IAEItemStack;
import appeng.client.texture.CableBusTextures;
import appeng.core.settings.TickRates;
import appeng.core.sync.packets.PacketTransitionEffect;
import appeng.hooks.TickHandler;
import appeng.me.GridAccessException;
import appeng.parts.PartBasicState;
import appeng.server.ServerHelper;
import appeng.util.Platform;
import appeng.util.item.AEItemStack;
import cpw.mods.fml.relauncher.Side;
import cpw.mods.fml.relauncher.SideOnly;

public class PartAnnihilationPlane extends PartBasicState implements IGridTickable, Callable
{

  public PartAnnihilationPlane(ItemStack is) {
    super( PartAnnihilationPlane.class, is );
  }

  @Override
  @SideOnly(Side.CLIENT)
  public void renderInventory(IPartRenderHelper rh, RenderBlocks renderer)
  {
    rh.setTexture( CableBusTextures.PartPlaneSides.getIcon(), CableBusTextures.PartPlaneSides.getIcon(),
        CableBusTextures.PartTransitionPlaneBack.getIcon(), is.getIconIndex(), CableBusTextures.PartPlaneSides.getIcon(),
        CableBusTextures.PartPlaneSides.getIcon() );

    rh.setBounds( 1, 1, 15, 15, 15, 16 );
    rh.renderInventoryBox( renderer );

    rh.setBounds( 5, 5, 14, 11, 11, 15 );
    rh.renderInventoryBox( renderer );
  }

  @Override
  @SideOnly(Side.CLIENT)
  public void renderStatic(int x, int y, int z, IPartRenderHelper rh, RenderBlocks renderer)
  {
    int minX = 1;
    int minY = 1;
    int maxX = 15;
    int maxY = 15;

    ForgeDirection e = rh.getWorldX();
    ForgeDirection u = rh.getWorldY();

    TileEntity te = getHost().getTile();

    if ( isTransitionPlane( te.getWorldObj().getTileEntity( x - e.offsetX, y - e.offsetY, z - e.offsetZ ), side ) )
      minX = 0;

    if ( isTransitionPlane( te.getWorldObj().getTileEntity( x + e.offsetX, y + e.offsetY, z + e.offsetZ ), side ) )
      maxX = 16;

    if ( isTransitionPlane( te.getWorldObj().getTileEntity( x - u.offsetX, y - u.offsetY, z - u.offsetZ ), side ) )
      minY = 0;

    if ( isTransitionPlane( te.getWorldObj().getTileEntity( x + u.offsetX, y + u.offsetY, z + u.offsetZ ), side ) )
      maxY = 16;

    boolean isActive = (clientFlags & (POWERED_FLAG | CHANNEL_FLAG)) == (POWERED_FLAG | CHANNEL_FLAG);

    renderCache = rh.useSimplifiedRendering( x, y, z, this, renderCache );
    rh.setTexture( CableBusTextures.PartPlaneSides.getIcon(), CableBusTextures.PartPlaneSides.getIcon(),
        CableBusTextures.PartTransitionPlaneBack.getIcon(), isActive ? CableBusTextures.BlockAnnihilationPlaneOn.getIcon() : is.getIconIndex(),
        CableBusTextures.PartPlaneSides.getIcon(), CableBusTextures.PartPlaneSides.getIcon() );

    rh.setBounds( minX, minY, 15, maxX, maxY, 16 );
    rh.renderBlock( x, y, z, renderer );

    rh.setTexture( CableBusTextures.PartMonitorSidesStatus.getIcon(), CableBusTextures.PartMonitorSidesStatus.getIcon(),
        CableBusTextures.PartTransitionPlaneBack.getIcon(), isActive ? CableBusTextures.BlockAnnihilationPlaneOn.getIcon() : is.getIconIndex(),
        CableBusTextures.PartMonitorSidesStatus.getIcon(), CableBusTextures.PartMonitorSidesStatus.getIcon() );

    rh.setBounds( 5, 5, 14, 11, 11, 15 );
    rh.renderBlock( x, y, z, renderer );

    renderLights( x, y, z, rh, renderer );
  }

  private boolean isTransitionPlane(TileEntity blockTileEntity, ForgeDirection side)
  {
    if ( blockTileEntity instanceof IPartHost )
    {
      IPart p = ((IPartHost) blockTileEntity).getPart( side );
      return p instanceof PartAnnihilationPlane;
    }
    return false;
  }

  @Override
  public void getBoxes(IPartCollisionHelper bch)
  {
    int minX = 1;
    int minY = 1;
    int maxX = 15;
    int maxY = 15;

    IPartHost host = getHost();
    if ( host != null )
    {
      TileEntity te = host.getTile();

      int x = te.xCoord;
      int y = te.yCoord;
      int z = te.zCoord;

      ForgeDirection e = bch.getWorldX();
      ForgeDirection u = bch.getWorldY();

      if ( isTransitionPlane( te.getWorldObj().getTileEntity( x - e.offsetX, y - e.offsetY, z - e.offsetZ ), side ) )
        minX = 0;

      if ( isTransitionPlane( te.getWorldObj().getTileEntity( x + e.offsetX, y + e.offsetY, z + e.offsetZ ), side ) )
        maxX = 16;

      if ( isTransitionPlane( te.getWorldObj().getTileEntity( x - u.offsetX, y - u.offsetY, z - u.offsetZ ), side ) )
        minY = 0;

      if ( isTransitionPlane( te.getWorldObj().getTileEntity( x + u.offsetX, y + u.offsetY, z + u.offsetZ ), side ) )
        maxY = 16;
    }

    bch.addBox( 5, 5, 14, 11, 11, 15 );
    bch.addBox( minX, minY, 15, maxX, maxY, bch.isBBCollision() ? 15 : 16 );
  }

  @Override
  public int cableConnectionRenderTo()
  {
    return 1;
  }

  boolean breaking = false;
  final LinkedList<IAEItemStack> Buffer = new LinkedList<IAEItemStack>();
  final BaseActionSource mySrc = new MachineSource( this );

  @Override
  public void writeToNBT(NBTTagCompound data)
  {
    super.writeToNBT( data );

    data.setInteger( "bufferSize", Buffer.size() );
    for (int x = 0; x < Buffer.size(); x++)
    {
      NBTTagCompound pack = new NBTTagCompound();
      Buffer.get( x ).writeToNBT( pack );
      data.setTag( "buffer" + x, pack );
    }
  }

  @Override
  public void readFromNBT(NBTTagCompound data)
  {
    super.readFromNBT( data );

    int size = data.getInteger( "bufferSize" );
    Buffer.clear();
    for (int x = 0; x < size; x++)
    {
      NBTTagCompound pack = (NBTTagCompound) data.getTag( "buffer" + x );
      IAEItemStack ais = AEItemStack.loadItemStackFromNBT( pack );
      if ( ais != null )
        Buffer.add( ais );
    }
  }

  private boolean isAccepting()
  {
    return Buffer.isEmpty();
  }

  @Override
  public void getDrops(List<ItemStack> drops, boolean wrenched)
  {
    for (IAEItemStack is : Buffer)
      if ( is != null )
        drops.add( is.getItemStack() );
  }

  @Override
  @MENetworkEventSubscribe
  public void chanRender(MENetworkChannelsChanged c)
  {
    onNeighborChanged();
    getHost().markForUpdate();
  }

  @Override
  @MENetworkEventSubscribe
  public void powerRender(MENetworkPowerStatusChange c)
  {
    onNeighborChanged();
    getHost().markForUpdate();
  }

  public TickRateModulation EatBlock(boolean eatForReal)
  {
    if ( isAccepting() && proxy.isActive() )
    {
      try
      {
        TileEntity te = getTile();
        WorldServer w = (WorldServer) te.getWorldObj();

        int x = te.xCoord + side.offsetX;
        int y = te.yCoord + side.offsetY;
        int z = te.zCoord + side.offsetZ;

        Block blk = w.getBlock( x, y, z );

        IStorageGrid storage = proxy.getStorage();
        IEnergyGrid energy = proxy.getEnergy();

        Material mat = blk.getMaterial();
        boolean ignore = mat == Material.air || mat == Material.lava || mat == Material.water || mat.isLiquid() || blk == Blocks.bedrock
            || blk == Blocks.end_portal || blk == Blocks.end_portal_frame || blk == Blocks.command_block;

        if ( !ignore )
        {
          if ( !w.isAirBlock( x, y, z ) && w.blockExists( x, y, z ) && w.canMineBlock( Platform.getPlayer( w ), x, y, z ) )
          {
            float hardness = blk.getBlockHardness( w, x, y, z );
            if ( hardness >= 0.0 )
            {
              ItemStack[] out = Platform.getBlockDrops( w, x, y, z );
              float total = 1 + hardness;
              for (ItemStack is : out)
                total += is.stackSize;

              boolean hasPower = energy.extractAEPower( total, Actionable.SIMULATE, PowerMultiplier.CONFIG ) > total - 0.1;
              if ( hasPower )
              {
                if ( eatForReal )
                {
                  energy.extractAEPower( total, Actionable.MODULATE, PowerMultiplier.CONFIG );
                  w.setBlock( x, y, z, Platform.air, 0, 3 );

                  AxisAlignedBB box = AxisAlignedBB.getBoundingBox( x - 0.2, y - 0.2, z - 0.2, x + 1.2, y + 1.2, z + 1.2 );
                  for (Object ei : w.getEntitiesWithinAABB( EntityItem.class, box ))
                  {
                    if ( ei instanceof EntityItem )
                    {
                      EntityItem item = (EntityItem) ei;
                      if ( !item.isDead )
                      {
                        IAEItemStack storedItem = AEItemStack.create( item.getEntityItem() );
                        storedItem = Platform.poweredInsert( energy, storage.getItemInventory(), storedItem, mySrc );
                        if ( storedItem != null )
                          Buffer.add( storedItem );

                        item.setDead();
                      }
                    }
                  }

                  ServerHelper.proxy.sendToAllNearExcept( null, x, y, z, 64, w, new PacketTransitionEffect( x, y, z, side, true ) );

                  for (ItemStack snaggedItem : out)
                  {
                    IAEItemStack storedItem = AEItemStack.create( snaggedItem );
                    storedItem = Platform.poweredInsert( energy, storage.getItemInventory(), storedItem, mySrc );
                    if ( storedItem != null )
                      Buffer.add( storedItem );
                  }

                  return TickRateModulation.URGENT;
                }
                else
                {
                  breaking = true;
                  TickHandler.instance.addCallable( this.tile.getWorldObj(), this );
                  return TickRateModulation.URGENT;
                }
              }
            }
          }
        }
      }
      catch (GridAccessException e1)
      {
        // :P
      }
    }

    // nothing to do here :)
    return TickRateModulation.SLEEP;
  }

  @Override
  public TickingRequest getTickingRequest(IGridNode node)
  {
    return new TickingRequest( TickRates.AnnihilationPlane.min, TickRates.AnnihilationPlane.max, false, true );
  }

  @Override
  public void onNeighborChanged()
  {
    try
    {
      proxy.getTick().alertDevice( proxy.getNode() );
    }
    catch (GridAccessException e)
    {
      // :P
    }
  }

  @Override
  public void onEntityCollision(Entity entity)
  {
    if ( entity instanceof EntityItem && !entity.isDead && isAccepting() )
    {
      boolean capture = false;

      switch (side)
      {
      case DOWN:
      case UP:
        if ( entity.posX > tile.xCoord && entity.posX < tile.xCoord + 1 )
          if ( entity.posZ > tile.zCoord && entity.posZ < tile.zCoord + 1 )
            if ( (entity.posY > tile.yCoord + 0.9 && side == ForgeDirection.UP) || (entity.posY < tile.yCoord + 0.1 && side == ForgeDirection.DOWN) )
              capture = true;
        break;
      case SOUTH:
      case NORTH:
        if ( entity.posX > tile.xCoord && entity.posX < tile.xCoord + 1 )
          if ( entity.posY > tile.yCoord && entity.posY < tile.yCoord + 1 )
            if ( (entity.posZ > tile.zCoord + 0.9 && side == ForgeDirection.SOUTH)
                || (entity.posZ < tile.zCoord + 0.1 && side == ForgeDirection.NORTH) )
              capture = true;
        break;
      case EAST:
      case WEST:
        if ( entity.posZ > tile.zCoord && entity.posZ < tile.zCoord + 1 )
          if ( entity.posY > tile.yCoord && entity.posY < tile.yCoord + 1 )
            if ( (entity.posX > tile.xCoord + 0.9 && side == ForgeDirection.EAST)
                || (entity.posX < tile.xCoord + 0.1 && side == ForgeDirection.WEST) )
              capture = true;
        break;
      default:
        // umm?
        break;
      }

      if ( capture && Platform.isServer() && proxy.isActive() )
      {
        IAEItemStack stack = AEItemStack.create( ((EntityItem) entity).getEntityItem() );
        if ( stack != null )
        {
          ServerHelper.proxy.sendToAllNearExcept( null, tile.xCoord, tile.yCoord, tile.zCoord, 64, tile.getWorldObj(),
              new PacketTransitionEffect( entity.posX, entity.posY, entity.posZ, side, false ) );

          Buffer.add( stack );
          storeBuffer();
          entity.setDead();
        }
      }
    }
  }

  @Override
  public TickRateModulation tickingRequest(IGridNode node, int TicksSinceLastCall)
  {
    if ( breaking )
      return TickRateModulation.URGENT;

    if ( isAccepting() )
      return EatBlock( false );
    else
    {
      storeBuffer();
      return TickRateModulation.IDLE;
    }
  }

  private void storeBuffer()
  {
    try
    {
      IStorageGrid storage = proxy.getStorage();
      IEnergyGrid energy = proxy.getEnergy();

      while (!Buffer.isEmpty())
      {
        IAEItemStack storedItem = Buffer.pop();
        storedItem = Platform.poweredInsert( energy, storage.getItemInventory(), storedItem, mySrc );
        if ( storedItem != null )
        {
          Buffer.add( storedItem );
          break;
        }
      }
    }
    catch (GridAccessException e1)
    {
      // :P
    }
  }

  @Override
  public Object call() throws Exception
  {
    breaking = false;
    return EatBlock( true );
  }

}
TOP

Related Classes of appeng.parts.automation.PartAnnihilationPlane

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.