478 lines
12 KiB
Java
478 lines
12 KiB
Java
package mekanism.common;
|
|
|
|
import ic2.api.energy.tile.IEnergySink;
|
|
|
|
import java.util.Arrays;
|
|
import java.util.Collection;
|
|
import java.util.Collections;
|
|
import java.util.HashSet;
|
|
import java.util.Iterator;
|
|
import java.util.List;
|
|
import java.util.Set;
|
|
|
|
import mekanism.api.ListUtils;
|
|
import mekanism.api.energy.IStrictEnergyAcceptor;
|
|
import mekanism.api.transmitters.DynamicNetwork;
|
|
import mekanism.api.transmitters.IGridTransmitter;
|
|
import mekanism.api.transmitters.TransmissionType;
|
|
import mekanism.api.transmitters.TransmitterNetworkRegistry;
|
|
import mekanism.common.util.CableUtils;
|
|
import mekanism.common.util.MekanismUtils;
|
|
import net.minecraft.tileentity.TileEntity;
|
|
import net.minecraftforge.common.ForgeDirection;
|
|
import net.minecraftforge.common.MinecraftForge;
|
|
import net.minecraftforge.event.Event;
|
|
import buildcraft.api.power.IPowerReceptor;
|
|
import buildcraft.api.power.PowerHandler.PowerReceiver;
|
|
import buildcraft.api.power.PowerHandler.Type;
|
|
import cofh.api.energy.IEnergyHandler;
|
|
import cpw.mods.fml.common.FMLCommonHandler;
|
|
|
|
public class EnergyNetwork extends DynamicNetwork<TileEntity, EnergyNetwork>
|
|
{
|
|
private double lastPowerScale = 0;
|
|
private double joulesTransmitted = 0;
|
|
private double jouleBufferLastTick = 0;
|
|
|
|
public double clientEnergyScale = 0;
|
|
|
|
public double electricityStored;
|
|
|
|
public EnergyNetwork(IGridTransmitter<EnergyNetwork>... varCables)
|
|
{
|
|
transmitters.addAll(Arrays.asList(varCables));
|
|
register();
|
|
}
|
|
|
|
public EnergyNetwork(Collection<IGridTransmitter<EnergyNetwork>> collection)
|
|
{
|
|
transmitters.addAll(collection);
|
|
register();
|
|
}
|
|
|
|
public static double round(double d)
|
|
{
|
|
return Math.round(d * 10000)/10000;
|
|
}
|
|
|
|
public EnergyNetwork(Set<EnergyNetwork> networks)
|
|
{
|
|
for(EnergyNetwork net : networks)
|
|
{
|
|
if(net != null)
|
|
{
|
|
if(net.jouleBufferLastTick > jouleBufferLastTick || net.clientEnergyScale > clientEnergyScale)
|
|
{
|
|
clientEnergyScale = net.clientEnergyScale;
|
|
jouleBufferLastTick = net.jouleBufferLastTick;
|
|
joulesTransmitted = net.joulesTransmitted;
|
|
lastPowerScale = net.lastPowerScale;
|
|
}
|
|
|
|
electricityStored += net.electricityStored;
|
|
|
|
addAllTransmitters(net.transmitters);
|
|
net.deregister();
|
|
}
|
|
}
|
|
|
|
refresh();
|
|
register();
|
|
}
|
|
|
|
@Override
|
|
public double getMeanCapacity()
|
|
{
|
|
//Use the harmonic mean. Because we're mean.
|
|
int numCables = transmitters.size();
|
|
double reciprocalSum = 0;
|
|
|
|
for(IGridTransmitter<EnergyNetwork> cable : transmitters)
|
|
{
|
|
reciprocalSum += 1.0/(double)cable.getCapacity();
|
|
}
|
|
|
|
return (double)numCables / reciprocalSum;
|
|
}
|
|
|
|
@Override
|
|
public void onNetworksCreated(List<EnergyNetwork> networks)
|
|
{
|
|
if(FMLCommonHandler.instance().getEffectiveSide().isServer())
|
|
{
|
|
double[] caps = new double[networks.size()];
|
|
double cap = 0;
|
|
|
|
for(EnergyNetwork network : networks)
|
|
{
|
|
caps[networks.indexOf(network)] = network.getCapacity();
|
|
cap += network.getCapacity();
|
|
}
|
|
|
|
electricityStored = Math.min(cap, electricityStored);
|
|
|
|
double[] percent = ListUtils.percent(caps);
|
|
|
|
for(EnergyNetwork network : networks)
|
|
{
|
|
network.electricityStored = round(percent[networks.indexOf(network)]*electricityStored);
|
|
}
|
|
}
|
|
}
|
|
|
|
public synchronized double getEnergyNeeded()
|
|
{
|
|
if(FMLCommonHandler.instance().getEffectiveSide().isClient())
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
return getCapacity()-electricityStored;
|
|
}
|
|
|
|
public synchronized double tickEmit(double energyToSend)
|
|
{
|
|
if(FMLCommonHandler.instance().getEffectiveSide().isClient())
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
double sent = 0;
|
|
boolean tryAgain = false;
|
|
|
|
do {
|
|
tryAgain = false;
|
|
|
|
double prev = sent;
|
|
sent += doEmit(energyToSend-sent);
|
|
|
|
if(energyToSend-sent > 0 && sent-prev > 0)
|
|
{
|
|
tryAgain = true;
|
|
}
|
|
} while(tryAgain);
|
|
|
|
joulesTransmitted = sent;
|
|
return sent;
|
|
}
|
|
|
|
public synchronized double emit(double energyToSend)
|
|
{
|
|
double toUse = Math.min(getEnergyNeeded(), energyToSend);
|
|
electricityStored += toUse;
|
|
return energyToSend-toUse;
|
|
}
|
|
|
|
/**
|
|
* @return sent
|
|
*/
|
|
public synchronized double doEmit(double energyToSend)
|
|
{
|
|
double sent = 0;
|
|
|
|
List availableAcceptors = Arrays.asList(getAcceptors().toArray());
|
|
|
|
Collections.shuffle(availableAcceptors);
|
|
|
|
if(!availableAcceptors.isEmpty())
|
|
{
|
|
int divider = availableAcceptors.size();
|
|
double remaining = energyToSend % divider;
|
|
double sending = (energyToSend-remaining)/divider;
|
|
|
|
for(Object obj : availableAcceptors)
|
|
{
|
|
if(obj instanceof TileEntity)
|
|
{
|
|
TileEntity acceptor = (TileEntity)obj;
|
|
double currentSending = sending+remaining;
|
|
ForgeDirection side = acceptorDirections.get(acceptor);
|
|
|
|
if(side == null)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
remaining = 0;
|
|
|
|
if(acceptor instanceof IStrictEnergyAcceptor)
|
|
{
|
|
sent += ((IStrictEnergyAcceptor)acceptor).transferEnergyToAcceptor(side.getOpposite(), currentSending);
|
|
}
|
|
else if(acceptor instanceof IEnergyHandler)
|
|
{
|
|
IEnergyHandler handler = (IEnergyHandler)acceptor;
|
|
int used = handler.receiveEnergy(side.getOpposite(), (int)Math.round(currentSending*Mekanism.TO_TE), false);
|
|
sent += used*Mekanism.FROM_TE;
|
|
}
|
|
else if(acceptor instanceof IEnergySink)
|
|
{
|
|
double toSend = Math.min(currentSending, ((IEnergySink)acceptor).getMaxSafeInput()*Mekanism.FROM_IC2);
|
|
toSend = Math.min(toSend, ((IEnergySink)acceptor).demandedEnergyUnits()*Mekanism.FROM_IC2);
|
|
sent += (toSend - (((IEnergySink)acceptor).injectEnergyUnits(side.getOpposite(), toSend*Mekanism.TO_IC2)*Mekanism.FROM_IC2));
|
|
}
|
|
else if(acceptor instanceof IPowerReceptor && MekanismUtils.useBuildCraft())
|
|
{
|
|
PowerReceiver receiver = ((IPowerReceptor)acceptor).getPowerReceiver(side.getOpposite());
|
|
|
|
if(receiver != null)
|
|
{
|
|
float toSend = receiver.receiveEnergy(Type.PIPE, (float)(Math.min(receiver.powerRequest(), currentSending*Mekanism.TO_BC)), side.getOpposite());
|
|
sent += toSend*Mekanism.FROM_BC;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return sent;
|
|
}
|
|
|
|
@Override
|
|
public synchronized Set<TileEntity> getAcceptors(Object... data)
|
|
{
|
|
Set<TileEntity> toReturn = new HashSet<TileEntity>();
|
|
|
|
if(FMLCommonHandler.instance().getEffectiveSide().isClient())
|
|
{
|
|
return toReturn;
|
|
}
|
|
|
|
Set<TileEntity> copy = (Set<TileEntity>)possibleAcceptors.clone();
|
|
|
|
for(TileEntity acceptor : copy)
|
|
{
|
|
ForgeDirection side = acceptorDirections.get(acceptor);
|
|
|
|
if(side == null)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if(acceptor instanceof IStrictEnergyAcceptor)
|
|
{
|
|
IStrictEnergyAcceptor handler = (IStrictEnergyAcceptor)acceptor;
|
|
|
|
if(handler.canReceiveEnergy(side.getOpposite()))
|
|
{
|
|
if(handler.getMaxEnergy() - handler.getEnergy() > 0)
|
|
{
|
|
toReturn.add(acceptor);
|
|
}
|
|
}
|
|
}
|
|
else if(acceptor instanceof IEnergyHandler)
|
|
{
|
|
IEnergyHandler handler = (IEnergyHandler)acceptor;
|
|
|
|
if(handler.canInterface(side.getOpposite()))
|
|
{
|
|
if(handler.getMaxEnergyStored(side.getOpposite()) - handler.getEnergyStored(side.getOpposite()) > 0)
|
|
{
|
|
toReturn.add(acceptor);
|
|
}
|
|
}
|
|
}
|
|
else if(acceptor instanceof IEnergySink)
|
|
{
|
|
IEnergySink handler = (IEnergySink)acceptor;
|
|
|
|
if(handler.acceptsEnergyFrom(null, side.getOpposite()))
|
|
{
|
|
if(Math.min((handler.demandedEnergyUnits()*Mekanism.FROM_IC2), (handler.getMaxSafeInput()*Mekanism.FROM_IC2)) > 0)
|
|
{
|
|
toReturn.add(acceptor);
|
|
}
|
|
}
|
|
}
|
|
else if(acceptor instanceof IPowerReceptor && MekanismUtils.useBuildCraft())
|
|
{
|
|
IPowerReceptor handler = (IPowerReceptor)acceptor;
|
|
|
|
if(handler.getPowerReceiver(side.getOpposite()) != null)
|
|
{
|
|
if((handler.getPowerReceiver(side.getOpposite()).powerRequest()*Mekanism.FROM_BC) > 0)
|
|
{
|
|
toReturn.add(acceptor);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return toReturn;
|
|
}
|
|
|
|
@Override
|
|
public synchronized void refresh()
|
|
{
|
|
Set<IGridTransmitter<EnergyNetwork>> iterCables = (Set<IGridTransmitter<EnergyNetwork>>)transmitters.clone();
|
|
Iterator<IGridTransmitter<EnergyNetwork>> it = iterCables.iterator();
|
|
|
|
possibleAcceptors.clear();
|
|
acceptorDirections.clear();
|
|
|
|
while(it.hasNext())
|
|
{
|
|
IGridTransmitter<EnergyNetwork> conductor = (IGridTransmitter<EnergyNetwork>)it.next();
|
|
|
|
if(conductor == null || ((TileEntity)conductor).isInvalid())
|
|
{
|
|
it.remove();
|
|
transmitters.remove(conductor);
|
|
}
|
|
else {
|
|
conductor.setTransmitterNetwork(this);
|
|
}
|
|
}
|
|
|
|
for(IGridTransmitter<EnergyNetwork> transmitter : iterCables)
|
|
{
|
|
TileEntity[] acceptors = CableUtils.getConnectedEnergyAcceptors((TileEntity)transmitter);
|
|
|
|
for(TileEntity acceptor : acceptors)
|
|
{
|
|
ForgeDirection side = ForgeDirection.getOrientation(Arrays.asList(acceptors).indexOf(acceptor));
|
|
|
|
if(side != null && acceptor != null && !(acceptor instanceof IGridTransmitter) && transmitter.canConnectToAcceptor(side, true))
|
|
{
|
|
possibleAcceptors.add(acceptor);
|
|
acceptorDirections.put(acceptor, ForgeDirection.getOrientation(Arrays.asList(acceptors).indexOf(acceptor)));
|
|
}
|
|
}
|
|
}
|
|
|
|
needsUpdate = true;
|
|
}
|
|
|
|
@Override
|
|
public synchronized void merge(EnergyNetwork network)
|
|
{
|
|
if(network != null && network != this)
|
|
{
|
|
Set<EnergyNetwork> networks = new HashSet<EnergyNetwork>();
|
|
networks.add(this);
|
|
networks.add(network);
|
|
EnergyNetwork newNetwork = create(networks);
|
|
newNetwork.refresh();
|
|
}
|
|
}
|
|
|
|
public static class EnergyTransferEvent extends Event
|
|
{
|
|
public final EnergyNetwork energyNetwork;
|
|
|
|
public final double power;
|
|
|
|
public EnergyTransferEvent(EnergyNetwork network, double currentPower)
|
|
{
|
|
energyNetwork = network;
|
|
power = currentPower;
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public String toString()
|
|
{
|
|
return "[EnergyNetwork] " + transmitters.size() + " transmitters, " + possibleAcceptors.size() + " acceptors.";
|
|
}
|
|
|
|
@Override
|
|
public void onUpdate()
|
|
{
|
|
super.onUpdate();
|
|
|
|
clearJoulesTransmitted();
|
|
|
|
double currentPowerScale = getPowerScale();
|
|
|
|
if(FMLCommonHandler.instance().getEffectiveSide().isServer())
|
|
{
|
|
if(Math.abs(currentPowerScale-lastPowerScale) > 0.01 || (currentPowerScale != lastPowerScale && (currentPowerScale == 0 || currentPowerScale == 1)))
|
|
{
|
|
needsUpdate = true;
|
|
}
|
|
|
|
if(needsUpdate)
|
|
{
|
|
MinecraftForge.EVENT_BUS.post(new EnergyTransferEvent(this, currentPowerScale));
|
|
lastPowerScale = currentPowerScale;
|
|
needsUpdate = false;
|
|
}
|
|
|
|
if(electricityStored > 0)
|
|
{
|
|
electricityStored -= tickEmit(electricityStored);
|
|
}
|
|
}
|
|
}
|
|
|
|
public double getPowerScale()
|
|
{
|
|
return Math.max(jouleBufferLastTick == 0 ? 0 : Math.min(Math.ceil(Math.log10(getPower())*2)/10, 1), getCapacity() == 0 ? 0 : electricityStored/getCapacity());
|
|
}
|
|
|
|
public void clearJoulesTransmitted()
|
|
{
|
|
jouleBufferLastTick = electricityStored;
|
|
joulesTransmitted = 0;
|
|
}
|
|
|
|
public double getPower()
|
|
{
|
|
return jouleBufferLastTick * 20;
|
|
}
|
|
|
|
@Override
|
|
protected EnergyNetwork create(IGridTransmitter<EnergyNetwork>... varTransmitters)
|
|
{
|
|
EnergyNetwork network = new EnergyNetwork(varTransmitters);
|
|
network.clientEnergyScale = clientEnergyScale;
|
|
network.jouleBufferLastTick = jouleBufferLastTick;
|
|
network.joulesTransmitted = joulesTransmitted;
|
|
network.lastPowerScale = lastPowerScale;
|
|
network.electricityStored += electricityStored;
|
|
return network;
|
|
}
|
|
|
|
@Override
|
|
protected EnergyNetwork create(Collection<IGridTransmitter<EnergyNetwork>> collection)
|
|
{
|
|
EnergyNetwork network = new EnergyNetwork(collection);
|
|
network.clientEnergyScale = clientEnergyScale;
|
|
network.jouleBufferLastTick = jouleBufferLastTick;
|
|
network.joulesTransmitted = joulesTransmitted;
|
|
network.lastPowerScale = lastPowerScale;
|
|
network.electricityStored += electricityStored;
|
|
return network;
|
|
}
|
|
|
|
@Override
|
|
protected EnergyNetwork create(Set<EnergyNetwork> networks)
|
|
{
|
|
return new EnergyNetwork(networks);
|
|
}
|
|
|
|
@Override
|
|
public TransmissionType getTransmissionType()
|
|
{
|
|
return TransmissionType.ENERGY;
|
|
}
|
|
|
|
@Override
|
|
public String getNeeded()
|
|
{
|
|
return MekanismUtils.getEnergyDisplay(getEnergyNeeded());
|
|
}
|
|
|
|
@Override
|
|
public String getStored()
|
|
{
|
|
return MekanismUtils.getEnergyDisplay(electricityStored);
|
|
}
|
|
|
|
@Override
|
|
public String getFlow()
|
|
{
|
|
return MekanismUtils.getPowerDisplay(20*joulesTransmitted);
|
|
}
|
|
}
|