/* * 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.me.cache; import java.util.HashMap; import java.util.LinkedList; import java.util.List; import appeng.api.networking.IGrid; import appeng.api.networking.IGridHost; import appeng.api.networking.IGridNode; import appeng.api.networking.IGridStorage; import appeng.api.networking.events.MENetworkBootingStatusChange; import appeng.api.networking.events.MENetworkEventSubscribe; import appeng.api.networking.spatial.ISpatialCache; import appeng.api.util.DimensionalCoord; import appeng.api.util.IReadOnlyCollection; import appeng.core.AEConfig; import appeng.me.cluster.implementations.SpatialPylonCluster; import appeng.tile.spatial.TileSpatialIOPort; import appeng.tile.spatial.TileSpatialPylon; public class SpatialPylonCache implements ISpatialCache { private final IGrid myGrid; private long powerRequired = 0; private double efficiency = 0.0; private DimensionalCoord captureMin; private DimensionalCoord captureMax; private boolean isValid = false; private List ioPorts = new LinkedList(); private HashMap clusters = new HashMap(); public SpatialPylonCache( final IGrid g ) { this.myGrid = g; } @MENetworkEventSubscribe public void bootingRender( final MENetworkBootingStatusChange c ) { this.reset( this.myGrid ); } private void reset( final IGrid grid ) { this.clusters = new HashMap(); this.ioPorts = new LinkedList(); for( final IGridNode gm : grid.getMachines( TileSpatialIOPort.class ) ) { this.ioPorts.add( (TileSpatialIOPort) gm.getMachine() ); } final IReadOnlyCollection set = grid.getMachines( TileSpatialPylon.class ); for( final IGridNode gm : set ) { if( gm.meetsChannelRequirements() ) { final SpatialPylonCluster c = ( (TileSpatialPylon) gm.getMachine() ).getCluster(); if( c != null ) { this.clusters.put( c, c ); } } } this.captureMax = null; this.captureMin = null; this.isValid = true; int pylonBlocks = 0; for( final SpatialPylonCluster cl : this.clusters.values() ) { if( this.captureMax == null ) { this.captureMax = cl.getMax().copy(); } if( this.captureMin == null ) { this.captureMin = cl.getMin().copy(); } pylonBlocks += cl.tileCount(); this.captureMin.x = Math.min( this.captureMin.x, cl.getMin().x ); this.captureMin.y = Math.min( this.captureMin.y, cl.getMin().y ); this.captureMin.z = Math.min( this.captureMin.z, cl.getMin().z ); this.captureMax.x = Math.max( this.captureMax.x, cl.getMax().x ); this.captureMax.y = Math.max( this.captureMax.y, cl.getMax().y ); this.captureMax.z = Math.max( this.captureMax.z, cl.getMax().z ); } double maxPower = 0; double minPower = 0; if( this.hasRegion() ) { this.isValid = this.captureMax.x - this.captureMin.x > 1 && this.captureMax.y - this.captureMin.y > 1 && this.captureMax.z - this.captureMin.z > 1; for( final SpatialPylonCluster cl : this.clusters.values() ) { switch( cl.getCurrentAxis() ) { case X: this.isValid = this.isValid && ( ( this.captureMax.y == cl.getMin().y || this.captureMin.y == cl.getMax().y ) || ( this.captureMax.z == cl.getMin().z || this.captureMin.z == cl.getMax().z ) ) && ( ( this.captureMax.y == cl.getMax().y || this.captureMin.y == cl.getMin().y ) || ( this.captureMax.z == cl.getMax().z || this.captureMin.z == cl.getMin().z ) ); break; case Y: this.isValid = this.isValid && ( ( this.captureMax.x == cl.getMin().x || this.captureMin.x == cl.getMax().x ) || ( this.captureMax.z == cl.getMin().z || this.captureMin.z == cl.getMax().z ) ) && ( ( this.captureMax.x == cl.getMax().x || this.captureMin.x == cl.getMin().x ) || ( this.captureMax.z == cl.getMax().z || this.captureMin.z == cl.getMin().z ) ); break; case Z: this.isValid = this.isValid && ( ( this.captureMax.y == cl.getMin().y || this.captureMin.y == cl.getMax().y ) || ( this.captureMax.x == cl.getMin().x || this.captureMin.x == cl.getMax().x ) ) && ( ( this.captureMax.y == cl.getMax().y || this.captureMin.y == cl.getMin().y ) || ( this.captureMax.x == cl.getMax().x || this.captureMin.x == cl.getMin().x ) ); break; case UNFORMED: this.isValid = false; break; } } final int reqX = this.captureMax.x - this.captureMin.x; final int reqY = this.captureMax.y - this.captureMin.y; final int reqZ = this.captureMax.z - this.captureMin.z; final int requirePylonBlocks = Math.max( 6, ( ( reqX * reqZ + reqX * reqY + reqY * reqZ ) * 3 ) / 8 ); this.efficiency = (double) pylonBlocks / (double) requirePylonBlocks; if( this.efficiency > 1.0 ) { this.efficiency = 1.0; } if( this.efficiency < 0.0 ) { this.efficiency = 0.0; } minPower = (double) reqX * (double) reqY * reqZ * AEConfig.instance.spatialPowerMultiplier; maxPower = Math.pow( minPower, AEConfig.instance.spatialPowerExponent ); } final double affective_efficiency = Math.pow( this.efficiency, 0.25 ); this.powerRequired = (long) ( affective_efficiency * minPower + ( 1.0 - affective_efficiency ) * maxPower ); for( final SpatialPylonCluster cl : this.clusters.values() ) { final boolean myWasValid = cl.isValid(); cl.setValid( this.isValid ); if( myWasValid != this.isValid ) { cl.updateStatus( false ); } } } @Override public boolean hasRegion() { return this.captureMin != null; } @Override public boolean isValidRegion() { return this.hasRegion() && this.isValid; } @Override public DimensionalCoord getMin() { return this.captureMin; } @Override public DimensionalCoord getMax() { return this.captureMax; } @Override public long requiredPower() { return this.powerRequired; } @Override public float currentEfficiency() { return (float) this.efficiency * 100; } @Override public void onUpdateTick() { } @Override public void removeNode( final IGridNode node, final IGridHost machine ) { } @Override public void addNode( final IGridNode node, final IGridHost machine ) { } @Override public void onSplit( final IGridStorage storageB ) { } @Override public void onJoin( final IGridStorage storageB ) { } @Override public void populateGridStorage( final IGridStorage storage ) { } }