Applied-Energistics-2-tiler.../src/main/java/appeng/client/render/BusRenderHelper.java

636 lines
15 KiB
Java
Raw Normal View History

2014-11-14 12:02:52 +01:00
/*
* This file is part of Applied Energistics 2.
* Copyright (c) 2013 - 2015, AlgorithmX2, All rights reserved.
2014-11-14 12:02:52 +01:00
*
* 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>.
*/
2014-09-24 02:26:27 +02:00
package appeng.client.render;
2014-09-24 02:26:27 +02:00
import java.util.EnumSet;
import javax.annotation.Nullable;
2014-09-24 02:26:27 +02:00
import com.google.common.base.Function;
import com.google.common.base.Optional;
2014-09-24 02:26:27 +02:00
import net.minecraft.block.Block;
import net.minecraft.client.renderer.RenderBlocks;
import net.minecraft.client.renderer.Tessellator;
import net.minecraft.util.IIcon;
import net.minecraftforge.common.util.ForgeDirection;
2014-12-29 21:59:05 +01:00
import cpw.mods.fml.relauncher.Side;
import cpw.mods.fml.relauncher.SideOnly;
2014-09-24 02:26:27 +02:00
import appeng.api.AEApi;
import appeng.api.exceptions.MissingDefinition;
2014-09-24 02:26:27 +02:00
import appeng.api.parts.IBoxProvider;
import appeng.api.parts.IPartCollisionHelper;
import appeng.api.parts.IPartRenderHelper;
import appeng.api.parts.ISimplifiedBundle;
import appeng.block.AEBaseBlock;
import appeng.block.networking.BlockCableBus;
import appeng.core.AEConfig;
import appeng.core.features.AEFeature;
@SideOnly( Side.CLIENT )
public final class BusRenderHelper implements IPartRenderHelper
2014-09-24 02:26:27 +02:00
{
public static final BusRenderHelper INSTANCE = new BusRenderHelper();
private static final int HEX_WHITE = 0xffffff;
private final BoundBoxCalculator bbc;
private final boolean noAlphaPass;
private final BaseBlockRender bbr;
private final Optional<Block> maybeBlock;
private final Optional<AEBaseBlock> maybeBaseBlock;
private int renderingForPass;
private int currentPass;
private int itemsRendered;
private double minX;
private double minY;
private double minZ;
private double maxX;
private double maxY;
private double maxZ;
private ForgeDirection ax;
private ForgeDirection ay;
private ForgeDirection az;
private int color;
public BusRenderHelper()
{
this.bbc = new BoundBoxCalculator( this );
this.noAlphaPass = !AEConfig.instance.isFeatureEnabled( AEFeature.AlphaPass );
this.bbr = new BaseBlockRender();
this.renderingForPass = 0;
this.currentPass = 0;
this.itemsRendered = 0;
this.minX = 0;
this.minY = 0;
this.minZ = 0;
this.maxX = 16;
this.maxY = 16;
this.maxZ = 16;
this.ax = ForgeDirection.EAST;
this.az = ForgeDirection.SOUTH;
this.ay = ForgeDirection.UP;
this.color = HEX_WHITE;
this.maybeBlock = AEApi.instance().definitions().blocks().multiPart().maybeBlock();
this.maybeBaseBlock = this.maybeBlock.transform( new BaseBlockTransformFunction() );
}
public int getItemsRendered()
{
return this.itemsRendered;
}
2014-09-24 02:26:27 +02:00
public void setPass( int pass )
{
this.renderingForPass = 0;
this.currentPass = pass;
this.itemsRendered = 0;
}
public double getBound( ForgeDirection side )
{
switch( side )
{
default:
case UNKNOWN:
return 0.5;
case DOWN:
return this.minY;
case EAST:
return this.maxX;
case NORTH:
return this.minZ;
case SOUTH:
return this.maxZ;
case UP:
return this.maxY;
case WEST:
return this.minX;
}
}
public void setRenderColor( int color )
{
for( Block block : AEApi.instance().definitions().blocks().multiPart().maybeBlock().asSet() )
{
final BlockCableBus cableBus = (BlockCableBus) block;
cableBus.setRenderColor( color );
}
}
public void setOrientation( ForgeDirection dx, ForgeDirection dy, ForgeDirection dz )
{
this.ax = dx == null ? ForgeDirection.EAST : dx;
this.ay = dy == null ? ForgeDirection.UP : dy;
this.az = dz == null ? ForgeDirection.SOUTH : dz;
}
2014-09-24 02:26:27 +02:00
public double[] getBounds()
{
return new double[] { this.minX, this.minY, this.minZ, this.maxX, this.maxY, this.maxZ };
}
2014-09-24 02:26:27 +02:00
public void setBounds( double[] bounds )
{
if( bounds == null || bounds.length != 6 )
2015-04-29 02:30:53 +02:00
{
return;
2015-04-29 02:30:53 +02:00
}
2014-09-24 02:26:27 +02:00
this.minX = bounds[0];
this.minY = bounds[1];
this.minZ = bounds[2];
this.maxX = bounds[3];
this.maxY = bounds[4];
this.maxZ = bounds[5];
}
2014-09-24 02:26:27 +02:00
private static class BoundBoxCalculator implements IPartCollisionHelper
2014-09-24 02:26:27 +02:00
{
private final BusRenderHelper helper;
private boolean started = false;
2014-09-24 02:26:27 +02:00
private float minX;
private float minY;
private float minZ;
2014-09-24 02:26:27 +02:00
private float maxX;
private float maxY;
private float maxZ;
2014-09-24 02:26:27 +02:00
public BoundBoxCalculator( BusRenderHelper helper )
{
this.helper = helper;
}
2014-09-24 02:26:27 +02:00
@Override
public void addBox( double minX, double minY, double minZ, double maxX, double maxY, double maxZ )
2014-09-24 02:26:27 +02:00
{
if( this.started )
2014-09-24 02:26:27 +02:00
{
this.minX = Math.min( this.minX, (float) minX );
this.minY = Math.min( this.minY, (float) minY );
this.minZ = Math.min( this.minZ, (float) minZ );
this.maxX = Math.max( this.maxX, (float) maxX );
this.maxY = Math.max( this.maxY, (float) maxY );
this.maxZ = Math.max( this.maxZ, (float) maxZ );
}
else
{
2014-12-29 15:13:47 +01:00
this.started = true;
2014-09-24 02:26:27 +02:00
this.minX = (float) minX;
this.minY = (float) minY;
this.minZ = (float) minZ;
this.maxX = (float) maxX;
this.maxY = (float) maxY;
this.maxZ = (float) maxZ;
}
}
@Override
public ForgeDirection getWorldX()
{
return this.helper.ax;
2014-09-24 02:26:27 +02:00
}
@Override
public ForgeDirection getWorldY()
{
return this.helper.ay;
2014-09-24 02:26:27 +02:00
}
@Override
public ForgeDirection getWorldZ()
{
return this.helper.az;
2014-09-24 02:26:27 +02:00
}
@Override
public boolean isBBCollision()
{
return false;
}
}
private static final class BaseBlockTransformFunction implements Function<Block, AEBaseBlock>
{
@Nullable
@Override
public AEBaseBlock apply( Block input )
{
if( input instanceof AEBaseBlock )
{
return ( (AEBaseBlock) input );
}
return null;
}
}
@Override
public void renderForPass( int pass )
2014-09-24 02:26:27 +02:00
{
this.renderingForPass = pass;
2014-09-24 02:26:27 +02:00
}
public boolean renderThis()
{
if( this.renderingForPass == this.currentPass || this.noAlphaPass )
2014-09-24 02:26:27 +02:00
{
2014-12-29 15:13:47 +01:00
this.itemsRendered++;
2014-09-24 02:26:27 +02:00
return true;
}
return false;
}
@Override
public void normalRendering()
{
2015-01-01 22:13:10 +01:00
RenderBlocksWorkaround rbw = BusRenderer.INSTANCE.renderer;
2014-09-24 02:26:27 +02:00
rbw.calculations = true;
rbw.useTextures = true;
rbw.enableAO = false;
}
@Override
public ISimplifiedBundle useSimplifiedRendering( int x, int y, int z, IBoxProvider p, ISimplifiedBundle sim )
2014-09-24 02:26:27 +02:00
{
2015-01-01 22:13:10 +01:00
RenderBlocksWorkaround rbw = BusRenderer.INSTANCE.renderer;
2014-09-24 02:26:27 +02:00
if( sim != null && this.maybeBlock.isPresent() && rbw.similarLighting( this.maybeBlock.get(), rbw.blockAccess, x, y, z, sim ) )
2014-09-24 02:26:27 +02:00
{
rbw.populate( sim );
rbw.faces = EnumSet.allOf( ForgeDirection.class );
rbw.calculations = false;
rbw.useTextures = false;
return sim;
}
else
{
boolean allFaces = rbw.renderAllFaces;
rbw.renderAllFaces = true;
rbw.calculations = true;
rbw.faces.clear();
2014-12-29 15:13:47 +01:00
this.bbc.started = false;
if( p == null )
2014-09-24 02:26:27 +02:00
{
2014-12-29 15:13:47 +01:00
this.bbc.minX = this.bbc.minY = this.bbc.minZ = 0;
this.bbc.maxX = this.bbc.maxY = this.bbc.maxZ = 16;
2014-09-24 02:26:27 +02:00
}
else
{
2014-12-29 15:13:47 +01:00
p.getBoxes( this.bbc );
if( this.bbc.minX < 1 )
2015-04-29 02:30:53 +02:00
{
2014-12-29 15:13:47 +01:00
this.bbc.minX = 1;
2015-04-29 02:30:53 +02:00
}
if( this.bbc.minY < 1 )
2015-04-29 02:30:53 +02:00
{
2014-12-29 15:13:47 +01:00
this.bbc.minY = 1;
2015-04-29 02:30:53 +02:00
}
if( this.bbc.minZ < 1 )
2015-04-29 02:30:53 +02:00
{
2014-12-29 15:13:47 +01:00
this.bbc.minZ = 1;
2015-04-29 02:30:53 +02:00
}
2014-12-29 15:13:47 +01:00
if( this.bbc.maxX > 15 )
2015-04-29 02:30:53 +02:00
{
2014-12-29 15:13:47 +01:00
this.bbc.maxX = 15;
2015-04-29 02:30:53 +02:00
}
if( this.bbc.maxY > 15 )
2015-04-29 02:30:53 +02:00
{
2014-12-29 15:13:47 +01:00
this.bbc.maxY = 15;
2015-04-29 02:30:53 +02:00
}
if( this.bbc.maxZ > 15 )
2015-04-29 02:30:53 +02:00
{
2014-12-29 15:13:47 +01:00
this.bbc.maxZ = 15;
2015-04-29 02:30:53 +02:00
}
2014-09-24 02:26:27 +02:00
}
2014-12-29 15:13:47 +01:00
this.setBounds( this.bbc.minX, this.bbc.minY, this.bbc.minZ, this.bbc.maxX, this.bbc.maxY, this.bbc.maxZ );
2014-09-24 02:26:27 +02:00
2014-12-29 15:13:47 +01:00
this.bbr.renderBlockBounds( rbw, this.minX, this.minY, this.minZ, this.maxX, this.maxY, this.maxZ, this.ax, this.ay, this.az );
for( Block block : this.maybeBlock.asSet() )
{
rbw.renderStandardBlock( block, x, y, z );
}
2014-09-24 02:26:27 +02:00
rbw.faces = EnumSet.allOf( ForgeDirection.class );
rbw.renderAllFaces = allFaces;
rbw.calculations = false;
rbw.useTextures = false;
return rbw.getLightingCache();
}
}
@Override
public void setBounds( float minX, float minY, float minZ, float maxX, float maxY, float maxZ )
2014-09-24 02:26:27 +02:00
{
this.minX = minX;
this.minY = minY;
this.minZ = minZ;
this.maxX = maxX;
this.maxY = maxY;
this.maxZ = maxZ;
2014-09-24 02:26:27 +02:00
}
@Override
public void setInvColor( int newColor )
2014-09-24 02:26:27 +02:00
{
2014-12-29 15:13:47 +01:00
this.color = newColor;
2014-09-24 02:26:27 +02:00
}
@Override
public void setTexture( IIcon ico )
2014-09-24 02:26:27 +02:00
{
for( AEBaseBlock baseBlock : this.maybeBaseBlock.asSet() )
{
baseBlock.getRendererInstance().setTemporaryRenderIcon( ico );
}
2014-09-24 02:26:27 +02:00
}
@Override
public void setTexture( IIcon down, IIcon up, IIcon north, IIcon south, IIcon west, IIcon east )
2014-09-24 02:26:27 +02:00
{
IIcon[] list = new IIcon[6];
2014-09-24 02:26:27 +02:00
list[0] = down;
list[1] = up;
list[2] = north;
list[3] = south;
list[4] = west;
list[5] = east;
2014-09-24 02:26:27 +02:00
for( AEBaseBlock baseBlock : this.maybeBaseBlock.asSet() )
{
baseBlock.getRendererInstance().setTemporaryRenderIcons( list[this.mapRotation( ForgeDirection.UP ).ordinal()], list[this.mapRotation( ForgeDirection.DOWN ).ordinal()], list[this.mapRotation( ForgeDirection.SOUTH ).ordinal()], list[this.mapRotation( ForgeDirection.NORTH ).ordinal()], list[this.mapRotation( ForgeDirection.EAST ).ordinal()], list[this.mapRotation( ForgeDirection.WEST ).ordinal()] );
}
2014-09-24 02:26:27 +02:00
}
public ForgeDirection mapRotation( ForgeDirection dir )
2014-09-24 02:26:27 +02:00
{
2014-12-29 15:13:47 +01:00
ForgeDirection forward = this.az;
ForgeDirection up = this.ay;
2014-09-24 02:26:27 +02:00
ForgeDirection west = ForgeDirection.UNKNOWN;
if( forward == null || up == null )
2015-04-29 02:30:53 +02:00
{
2014-09-24 02:26:27 +02:00
return dir;
2015-04-29 02:30:53 +02:00
}
2014-09-24 02:26:27 +02:00
int west_x = forward.offsetY * up.offsetZ - forward.offsetZ * up.offsetY;
int west_y = forward.offsetZ * up.offsetX - forward.offsetX * up.offsetZ;
int west_z = forward.offsetX * up.offsetY - forward.offsetY * up.offsetX;
for( ForgeDirection dx : ForgeDirection.VALID_DIRECTIONS )
2015-04-29 02:30:53 +02:00
{
if( dx.offsetX == west_x && dx.offsetY == west_y && dx.offsetZ == west_z )
2015-04-29 02:30:53 +02:00
{
2014-09-24 02:26:27 +02:00
west = dx;
2015-04-29 02:30:53 +02:00
}
}
2014-09-24 02:26:27 +02:00
if( dir == forward )
2015-04-29 02:30:53 +02:00
{
2014-09-24 02:26:27 +02:00
return ForgeDirection.SOUTH;
2015-04-29 02:30:53 +02:00
}
if( dir == forward.getOpposite() )
2015-04-29 02:30:53 +02:00
{
2014-09-24 02:26:27 +02:00
return ForgeDirection.NORTH;
2015-04-29 02:30:53 +02:00
}
2014-09-24 02:26:27 +02:00
if( dir == up )
2015-04-29 02:30:53 +02:00
{
2014-09-24 02:26:27 +02:00
return ForgeDirection.UP;
2015-04-29 02:30:53 +02:00
}
if( dir == up.getOpposite() )
2015-04-29 02:30:53 +02:00
{
2014-09-24 02:26:27 +02:00
return ForgeDirection.DOWN;
2015-04-29 02:30:53 +02:00
}
2014-09-24 02:26:27 +02:00
if( dir == west )
2015-04-29 02:30:53 +02:00
{
2014-09-24 02:26:27 +02:00
return ForgeDirection.WEST;
2015-04-29 02:30:53 +02:00
}
if( dir == west.getOpposite() )
2015-04-29 02:30:53 +02:00
{
2014-09-24 02:26:27 +02:00
return ForgeDirection.EAST;
2015-04-29 02:30:53 +02:00
}
2014-09-24 02:26:27 +02:00
return ForgeDirection.UNKNOWN;
}
@Override
public void renderInventoryBox( RenderBlocks renderer )
2014-09-24 02:26:27 +02:00
{
2014-12-29 15:13:47 +01:00
renderer.setRenderBounds( this.minX / 16.0, this.minY / 16.0, this.minZ / 16.0, this.maxX / 16.0, this.maxY / 16.0, this.maxZ / 16.0 );
for( AEBaseBlock baseBlock : this.maybeBaseBlock.asSet() )
{
this.bbr.renderInvBlock( EnumSet.allOf( ForgeDirection.class ), baseBlock, null, Tessellator.instance, this.color, renderer );
}
2014-09-24 02:26:27 +02:00
}
@Override
public void renderInventoryFace( IIcon icon, ForgeDirection face, RenderBlocks renderer )
2014-09-24 02:26:27 +02:00
{
2014-12-29 15:13:47 +01:00
renderer.setRenderBounds( this.minX / 16.0, this.minY / 16.0, this.minZ / 16.0, this.maxX / 16.0, this.maxY / 16.0, this.maxZ / 16.0 );
this.setTexture( icon );
for( AEBaseBlock baseBlock : this.maybeBaseBlock.asSet() )
{
this.bbr.renderInvBlock( EnumSet.of( face ), baseBlock, null, Tessellator.instance, this.color, renderer );
}
2014-09-24 02:26:27 +02:00
}
@Override
public void renderBlock( int x, int y, int z, RenderBlocks renderer )
2014-09-24 02:26:27 +02:00
{
if( !this.renderThis() )
2015-04-29 02:30:53 +02:00
{
2014-09-24 02:26:27 +02:00
return;
2015-04-29 02:30:53 +02:00
}
2014-09-24 02:26:27 +02:00
for( Block multiPart : AEApi.instance().definitions().blocks().multiPart().maybeBlock().asSet() )
{
final AEBaseBlock block = (AEBaseBlock) multiPart;
2014-09-24 02:26:27 +02:00
BlockRenderInfo info = block.getRendererInstance();
ForgeDirection forward = BusRenderHelper.INSTANCE.az;
ForgeDirection up = BusRenderHelper.INSTANCE.ay;
2014-09-24 02:26:27 +02:00
renderer.uvRotateBottom = info.getTexture( ForgeDirection.DOWN ).setFlip( BaseBlockRender.getOrientation( ForgeDirection.DOWN, forward, up ) );
renderer.uvRotateTop = info.getTexture( ForgeDirection.UP ).setFlip( BaseBlockRender.getOrientation( ForgeDirection.UP, forward, up ) );
2014-09-24 02:26:27 +02:00
renderer.uvRotateEast = info.getTexture( ForgeDirection.EAST ).setFlip( BaseBlockRender.getOrientation( ForgeDirection.EAST, forward, up ) );
renderer.uvRotateWest = info.getTexture( ForgeDirection.WEST ).setFlip( BaseBlockRender.getOrientation( ForgeDirection.WEST, forward, up ) );
2014-09-24 02:26:27 +02:00
renderer.uvRotateNorth = info.getTexture( ForgeDirection.NORTH ).setFlip( BaseBlockRender.getOrientation( ForgeDirection.NORTH, forward, up ) );
renderer.uvRotateSouth = info.getTexture( ForgeDirection.SOUTH ).setFlip( BaseBlockRender.getOrientation( ForgeDirection.SOUTH, forward, up ) );
this.bbr.renderBlockBounds( renderer, this.minX, this.minY, this.minZ, this.maxX, this.maxY, this.maxZ, this.ax, this.ay, this.az );
2014-09-24 02:26:27 +02:00
renderer.renderStandardBlock( block, x, y, z );
}
2014-09-24 02:26:27 +02:00
}
@Override
public Block getBlock()
{
for( Block block : AEApi.instance().definitions().blocks().multiPart().maybeBlock().asSet() )
{
return block;
}
2014-09-24 02:26:27 +02:00
throw new MissingDefinition( "Tried to access the multi part block without it being defined." );
2014-09-24 02:26:27 +02:00
}
public void prepareBounds( RenderBlocks renderer )
2014-09-24 02:26:27 +02:00
{
2014-12-29 15:13:47 +01:00
this.bbr.renderBlockBounds( renderer, this.minX, this.minY, this.minZ, this.maxX, this.maxY, this.maxZ, this.ax, this.ay, this.az );
2014-09-24 02:26:27 +02:00
}
@Override
public void setFacesToRender( EnumSet<ForgeDirection> faces )
2014-09-24 02:26:27 +02:00
{
2015-01-01 22:13:10 +01:00
BusRenderer.INSTANCE.renderer.renderFaces = faces;
2014-09-24 02:26:27 +02:00
}
@Override
public void renderBlockCurrentBounds( int x, int y, int z, RenderBlocks renderer )
2014-09-24 02:26:27 +02:00
{
if( !this.renderThis() )
2015-04-29 02:30:53 +02:00
{
2014-09-24 02:26:27 +02:00
return;
2015-04-29 02:30:53 +02:00
}
2014-09-24 02:26:27 +02:00
for( Block block : this.maybeBlock.asSet() )
{
renderer.renderStandardBlock( block, x, y, z );
}
2014-09-24 02:26:27 +02:00
}
@Override
public void renderFaceCutout( int x, int y, int z, IIcon ico, ForgeDirection face, float edgeThickness, RenderBlocks renderer )
2014-09-24 02:26:27 +02:00
{
if( !this.renderThis() )
2015-04-29 02:30:53 +02:00
{
2014-09-24 02:26:27 +02:00
return;
2015-04-29 02:30:53 +02:00
}
2014-09-24 02:26:27 +02:00
switch( face )
2014-09-24 02:26:27 +02:00
{
case DOWN:
face = this.ay.getOpposite();
break;
case EAST:
face = this.ax;
break;
case NORTH:
face = this.az.getOpposite();
break;
case SOUTH:
face = this.az;
break;
case UP:
face = this.ay;
break;
case WEST:
face = this.ax.getOpposite();
break;
case UNKNOWN:
break;
default:
break;
2014-09-24 02:26:27 +02:00
}
for( Block block : this.maybeBlock.asSet() )
{
this.bbr.renderCutoutFace( block, ico, x, y, z, renderer, face, edgeThickness );
}
2014-09-24 02:26:27 +02:00
}
@Override
public void renderFace( int x, int y, int z, IIcon ico, ForgeDirection face, RenderBlocks renderer )
2014-09-24 02:26:27 +02:00
{
if( !this.renderThis() )
2015-04-29 02:30:53 +02:00
{
2014-09-24 02:26:27 +02:00
return;
2015-04-29 02:30:53 +02:00
}
2014-09-24 02:26:27 +02:00
2014-12-29 15:13:47 +01:00
this.prepareBounds( renderer );
switch( face )
2014-09-24 02:26:27 +02:00
{
case DOWN:
face = this.ay.getOpposite();
break;
case EAST:
face = this.ax;
break;
case NORTH:
face = this.az.getOpposite();
break;
case SOUTH:
face = this.az;
break;
case UP:
face = this.ay;
break;
case WEST:
face = this.ax.getOpposite();
break;
case UNKNOWN:
break;
default:
break;
2014-09-24 02:26:27 +02:00
}
for( Block block : this.maybeBlock.asSet() )
{
this.bbr.renderFace( x, y, z, block, ico, renderer, face );
}
2014-09-24 02:26:27 +02:00
}
@Override
public ForgeDirection getWorldX()
{
2014-12-29 15:13:47 +01:00
return this.ax;
2014-09-24 02:26:27 +02:00
}
@Override
public ForgeDirection getWorldY()
{
2014-12-29 15:13:47 +01:00
return this.ay;
2014-09-24 02:26:27 +02:00
}
@Override
public ForgeDirection getWorldZ()
{
2014-12-29 15:13:47 +01:00
return this.az;
2014-09-24 02:26:27 +02:00
}
}