Improved transporter beacon usability
- add transporter room state to beacon - resume transport when conflicting beacons are removed - explicitly load chunks on beacon transport so players don't need their own - support transport inside non-celestial object dimension (wip)
This commit is contained in:
parent
a0b2c005d1
commit
f04782dd72
4 changed files with 146 additions and 76 deletions
|
@ -22,6 +22,7 @@ import net.minecraft.util.EnumFacing;
|
||||||
import net.minecraft.util.EnumHand;
|
import net.minecraft.util.EnumHand;
|
||||||
import net.minecraft.util.math.AxisAlignedBB;
|
import net.minecraft.util.math.AxisAlignedBB;
|
||||||
import net.minecraft.util.math.BlockPos;
|
import net.minecraft.util.math.BlockPos;
|
||||||
|
import net.minecraft.util.text.TextComponentString;
|
||||||
import net.minecraft.util.text.TextComponentTranslation;
|
import net.minecraft.util.text.TextComponentTranslation;
|
||||||
import net.minecraft.world.IBlockAccess;
|
import net.minecraft.world.IBlockAccess;
|
||||||
import net.minecraft.world.World;
|
import net.minecraft.world.World;
|
||||||
|
@ -168,9 +169,13 @@ public class BlockTransporterBeacon extends BlockAbstractContainer {
|
||||||
}
|
}
|
||||||
final TileEntityTransporterBeacon tileEntityTransporterBeacon = (TileEntityTransporterBeacon) tileEntity;
|
final TileEntityTransporterBeacon tileEntityTransporterBeacon = (TileEntityTransporterBeacon) tileEntity;
|
||||||
|
|
||||||
// sneaking with an empty hand
|
if (itemStackHeld.isEmpty()) {
|
||||||
if ( itemStackHeld.isEmpty()
|
if (!entityPlayer.isSneaking()) {// non-sneaking with an empty hand
|
||||||
&& entityPlayer.isSneaking() ) {
|
Commons.addChatMessage(entityPlayer, Commons.getChatPrefix(this)
|
||||||
|
.appendSibling(new TextComponentString(tileEntityTransporterBeacon.stateTransporter)));
|
||||||
|
return true;
|
||||||
|
|
||||||
|
} else {// sneaking with an empty hand
|
||||||
final boolean isEnabledOld = tileEntityTransporterBeacon.getIsEnabled();
|
final boolean isEnabledOld = tileEntityTransporterBeacon.getIsEnabled();
|
||||||
tileEntityTransporterBeacon.setIsEnabled(!isEnabledOld);
|
tileEntityTransporterBeacon.setIsEnabled(!isEnabledOld);
|
||||||
final boolean isEnabledNew = tileEntityTransporterBeacon.getIsEnabled();
|
final boolean isEnabledNew = tileEntityTransporterBeacon.getIsEnabled();
|
||||||
|
@ -185,6 +190,7 @@ public class BlockTransporterBeacon extends BlockAbstractContainer {
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return super.onBlockActivated(world, blockPos, blockState, entityPlayer, enumHand, enumFacing, hitX, hitY, hitZ);
|
return super.onBlockActivated(world, blockPos, blockState, entityPlayer, enumHand, enumFacing, hitX, hitY, hitZ);
|
||||||
}
|
}
|
||||||
|
|
|
@ -206,37 +206,6 @@ public class ItemBlockTransporterBeacon extends ItemBlockAbstractBase implements
|
||||||
super.onUpdate(itemStack, world, entity, indexSlot, isHeld);
|
super.onUpdate(itemStack, world, entity, indexSlot, isHeld);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* @TODO 1.10 is the upstream 1.7.10 bug fixed?
|
|
||||||
@Override
|
|
||||||
public boolean onItemUseFirst(final ItemStack itemStack, final EntityPlayer entityPlayer,
|
|
||||||
final World world, final int x, final int y, final int z,
|
|
||||||
final int side, final float hitX, final float hitY, final float hitZ) {
|
|
||||||
// itemStack is constantly updated for energy updates
|
|
||||||
// in net.minecraft.network.NetHandlerPlayServer.processPlayerBlockPlacement(NetHandlerPlayServer.java:657), a NPE appears randomly due to bad multithreading in upstream
|
|
||||||
// consequently, we prevent to use the item on any tile entity other than a TransporterCore
|
|
||||||
|
|
||||||
// allows block placement while sneaking
|
|
||||||
if (entityPlayer.isSneaking()) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// allows non-tile entities or transporter core
|
|
||||||
final TileEntity tileEntity = world.getTileEntity(x, y, z);
|
|
||||||
if ( tileEntity == null
|
|
||||||
|| tileEntity instanceof ITransporterCore ) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// allow if beacon is disabled
|
|
||||||
if (!isActive(itemStack)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// forbid everything else
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
/**/
|
|
||||||
|
|
||||||
@Nonnull
|
@Nonnull
|
||||||
@Override
|
@Override
|
||||||
public EnumActionResult onItemUse(@Nonnull final EntityPlayer entityPlayer,
|
public EnumActionResult onItemUse(@Nonnull final EntityPlayer entityPlayer,
|
||||||
|
|
|
@ -36,8 +36,9 @@ public class TileEntityTransporterBeacon extends TileEntityAbstractEnergyConsume
|
||||||
private UUID uuidTransporterCore;
|
private UUID uuidTransporterCore;
|
||||||
private int tickDeploying = 0;
|
private int tickDeploying = 0;
|
||||||
|
|
||||||
// computer properties
|
// computed properties
|
||||||
private boolean isActive = false;
|
private boolean isActive = false;
|
||||||
|
protected String stateTransporter = "";
|
||||||
|
|
||||||
public TileEntityTransporterBeacon() {
|
public TileEntityTransporterBeacon() {
|
||||||
super();
|
super();
|
||||||
|
@ -128,7 +129,11 @@ public class TileEntityTransporterBeacon extends TileEntityAbstractEnergyConsume
|
||||||
}
|
}
|
||||||
|
|
||||||
final TileEntityTransporterCore tileEntityTransporterCore = (TileEntityTransporterCore) tileEntity;
|
final TileEntityTransporterCore tileEntityTransporterCore = (TileEntityTransporterCore) tileEntity;
|
||||||
return tileEntityTransporterCore.updateBeacon(this, uuidTransporterCore);
|
final boolean isActive = tileEntityTransporterCore.updateBeacon(this, uuidTransporterCore);
|
||||||
|
final Object[] state = tileEntityTransporterCore.state();
|
||||||
|
stateTransporter = (String) state[1];
|
||||||
|
|
||||||
|
return isActive;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -144,6 +149,8 @@ public class TileEntityTransporterBeacon extends TileEntityAbstractEnergyConsume
|
||||||
if (isEnabled) {
|
if (isEnabled) {
|
||||||
tickDeploying = 0;
|
tickDeploying = 0;
|
||||||
}
|
}
|
||||||
|
// always clear status
|
||||||
|
stateTransporter = "";
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -49,7 +49,6 @@ import net.minecraft.block.Block;
|
||||||
import net.minecraft.block.BlockLiquid;
|
import net.minecraft.block.BlockLiquid;
|
||||||
import net.minecraft.block.state.IBlockState;
|
import net.minecraft.block.state.IBlockState;
|
||||||
import net.minecraft.entity.Entity;
|
import net.minecraft.entity.Entity;
|
||||||
import net.minecraft.entity.EntityList;
|
|
||||||
import net.minecraft.entity.EntityLivingBase;
|
import net.minecraft.entity.EntityLivingBase;
|
||||||
import net.minecraft.entity.player.EntityPlayer;
|
import net.minecraft.entity.player.EntityPlayer;
|
||||||
import net.minecraft.entity.player.EntityPlayerMP;
|
import net.minecraft.entity.player.EntityPlayerMP;
|
||||||
|
@ -68,10 +67,14 @@ import net.minecraft.util.EnumHand;
|
||||||
import net.minecraft.util.math.AxisAlignedBB;
|
import net.minecraft.util.math.AxisAlignedBB;
|
||||||
import net.minecraft.util.math.BlockPos;
|
import net.minecraft.util.math.BlockPos;
|
||||||
import net.minecraft.util.math.BlockPos.MutableBlockPos;
|
import net.minecraft.util.math.BlockPos.MutableBlockPos;
|
||||||
|
import net.minecraft.util.math.ChunkPos;
|
||||||
import net.minecraft.world.World;
|
import net.minecraft.world.World;
|
||||||
import net.minecraft.world.WorldServer;
|
import net.minecraft.world.WorldServer;
|
||||||
|
|
||||||
import net.minecraftforge.common.DimensionManager;
|
import net.minecraftforge.common.DimensionManager;
|
||||||
|
import net.minecraftforge.common.ForgeChunkManager;
|
||||||
|
import net.minecraftforge.common.ForgeChunkManager.Ticket;
|
||||||
|
import net.minecraftforge.common.ForgeChunkManager.Type;
|
||||||
import net.minecraftforge.common.util.Constants;
|
import net.minecraftforge.common.util.Constants;
|
||||||
import net.minecraftforge.fml.common.Optional;
|
import net.minecraftforge.fml.common.Optional;
|
||||||
|
|
||||||
|
@ -98,6 +101,7 @@ public class TileEntityTransporterCore extends TileEntityAbstractEnergyConsumer
|
||||||
private int tickComputerPulse = 0;
|
private int tickComputerPulse = 0;
|
||||||
private boolean isConnected = false;
|
private boolean isConnected = false;
|
||||||
private GlobalPosition globalPositionBeacon = null;
|
private GlobalPosition globalPositionBeacon = null;
|
||||||
|
private Ticket ticketChunkloading;
|
||||||
private double energyCostForAcquiring = 0.0D;
|
private double energyCostForAcquiring = 0.0D;
|
||||||
private double energyCostForEnergizing = 0.0D;
|
private double energyCostForEnergizing = 0.0D;
|
||||||
private double lockStrengthOptimal = -1.0D;
|
private double lockStrengthOptimal = -1.0D;
|
||||||
|
@ -258,6 +262,7 @@ public class TileEntityTransporterCore extends TileEntityAbstractEnergyConsumer
|
||||||
// execute state transitions
|
// execute state transitions
|
||||||
switch (transporterState) {
|
switch (transporterState) {
|
||||||
case DISABLED:
|
case DISABLED:
|
||||||
|
releaseChunks();
|
||||||
isLockRequested = false;
|
isLockRequested = false;
|
||||||
isEnergizeRequested = false;
|
isEnergizeRequested = false;
|
||||||
lockStrengthActual = 0.0D;
|
lockStrengthActual = 0.0D;
|
||||||
|
@ -272,6 +277,8 @@ public class TileEntityTransporterCore extends TileEntityAbstractEnergyConsumer
|
||||||
// force parameters validation for next tick
|
// force parameters validation for next tick
|
||||||
tickUpdateParameters = 0;
|
tickUpdateParameters = 0;
|
||||||
transporterState = EnumTransporterState.ACQUIRING;
|
transporterState = EnumTransporterState.ACQUIRING;
|
||||||
|
} else {
|
||||||
|
releaseChunks();
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
@ -321,6 +328,13 @@ public class TileEntityTransporterCore extends TileEntityAbstractEnergyConsumer
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onChunkUnload() {
|
||||||
|
releaseChunks();
|
||||||
|
|
||||||
|
super.onChunkUnload();
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void invalidate() {
|
public void invalidate() {
|
||||||
if (!world.isRemote) {
|
if (!world.isRemote) {
|
||||||
|
@ -340,6 +354,7 @@ public class TileEntityTransporterCore extends TileEntityAbstractEnergyConsumer
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
releaseChunks();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void state_energizing() {
|
private void state_energizing() {
|
||||||
|
@ -667,6 +682,30 @@ public class TileEntityTransporterCore extends TileEntityAbstractEnergyConsumer
|
||||||
long mass;
|
long mass;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void checkBeaconObsolescence() {
|
||||||
|
if (globalPositionBeacon != null) {
|
||||||
|
final WorldServer worldBeacon = globalPositionBeacon.getWorldServerIfLoaded();
|
||||||
|
if (worldBeacon == null) {
|
||||||
|
globalPositionBeacon = null;
|
||||||
|
isLockRequested = false;
|
||||||
|
isEnergizeRequested = false;
|
||||||
|
isJammed = false;
|
||||||
|
reasonJammed = "";
|
||||||
|
|
||||||
|
} else {
|
||||||
|
final TileEntity tileEntity = worldBeacon.getTileEntity(new BlockPos(globalPositionBeacon.x, globalPositionBeacon.y, globalPositionBeacon.z));
|
||||||
|
if ( !(tileEntity instanceof ITransporterBeacon)
|
||||||
|
|| !((ITransporterBeacon) tileEntity).isActive() ) {
|
||||||
|
globalPositionBeacon = null;
|
||||||
|
isLockRequested = false;
|
||||||
|
isEnergizeRequested = false;
|
||||||
|
isJammed = false;
|
||||||
|
reasonJammed = "";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private void updateParameters() {
|
private void updateParameters() {
|
||||||
isJammed = false;
|
isJammed = false;
|
||||||
reasonJammed = "";
|
reasonJammed = "";
|
||||||
|
@ -688,23 +727,8 @@ public class TileEntityTransporterCore extends TileEntityAbstractEnergyConsumer
|
||||||
final CelestialObject celestialObjectLocal = CelestialObjectManager.get(world, pos.getX(), pos.getZ());
|
final CelestialObject celestialObjectLocal = CelestialObjectManager.get(world, pos.getX(), pos.getZ());
|
||||||
final Vector3 v3Local_universal = StarMapRegistry.getUniversalCoordinates(celestialObjectLocal, globalPositionLocal.x, globalPositionLocal.y, globalPositionLocal.z);
|
final Vector3 v3Local_universal = StarMapRegistry.getUniversalCoordinates(celestialObjectLocal, globalPositionLocal.x, globalPositionLocal.y, globalPositionLocal.z);
|
||||||
|
|
||||||
// check beacon obsolescence
|
// validate context
|
||||||
if (globalPositionBeacon != null) {
|
checkBeaconObsolescence();
|
||||||
final WorldServer worldBeacon = globalPositionBeacon.getWorldServerIfLoaded();
|
|
||||||
if (worldBeacon == null) {
|
|
||||||
globalPositionBeacon = null;
|
|
||||||
isLockRequested = false;
|
|
||||||
isEnergizeRequested = false;
|
|
||||||
} else {
|
|
||||||
final TileEntity tileEntity = worldBeacon.getTileEntity(new BlockPos(globalPositionBeacon.x, globalPositionBeacon.y, globalPositionBeacon.z));
|
|
||||||
if ( !(tileEntity instanceof ITransporterBeacon)
|
|
||||||
|| !((ITransporterBeacon) tileEntity).isActive() ) {
|
|
||||||
globalPositionBeacon = null;
|
|
||||||
isLockRequested = false;
|
|
||||||
isEnergizeRequested = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// compute remote global position
|
// compute remote global position
|
||||||
GlobalPosition globalPositionRemoteNew = null;
|
GlobalPosition globalPositionRemoteNew = null;
|
||||||
|
@ -722,8 +746,12 @@ public class TileEntityTransporterCore extends TileEntityAbstractEnergyConsumer
|
||||||
globalPositionRemoteNew = new GlobalPosition(celestialObjectChild.dimensionId, vRequest.x, (vRequest.y + 1024) % 256, vRequest.z);
|
globalPositionRemoteNew = new GlobalPosition(celestialObjectChild.dimensionId, vRequest.x, (vRequest.y + 1024) % 256, vRequest.z);
|
||||||
}
|
}
|
||||||
} else if (vRequest.y > 256) {
|
} else if (vRequest.y > 256) {
|
||||||
|
if (celestialObjectLocal == null) {
|
||||||
|
reasonJammed = "Unknown dimension, no reachable orbit";
|
||||||
|
} else {
|
||||||
vRequest.translateBack(celestialObjectLocal.getEntryOffset());
|
vRequest.translateBack(celestialObjectLocal.getEntryOffset());
|
||||||
globalPositionRemoteNew = new GlobalPosition(celestialObjectLocal.parent.dimensionId, vRequest.x, vRequest.y % 256, vRequest.z);
|
globalPositionRemoteNew = new GlobalPosition(celestialObjectLocal.parent.dimensionId, vRequest.x, vRequest.y % 256, vRequest.z);
|
||||||
|
}
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
globalPositionRemoteNew = new GlobalPosition(world.provider.getDimension(), vRequest.x, vRequest.y, vRequest.z);
|
globalPositionRemoteNew = new GlobalPosition(world.provider.getDimension(), vRequest.x, vRequest.y, vRequest.z);
|
||||||
|
@ -783,7 +811,8 @@ public class TileEntityTransporterCore extends TileEntityAbstractEnergyConsumer
|
||||||
|
|
||||||
// validate cross dimension transport rules
|
// validate cross dimension transport rules
|
||||||
if (celestialObjectLocal != celestialObjectRemote) {
|
if (celestialObjectLocal != celestialObjectRemote) {
|
||||||
if ( celestialObjectLocal.isHyperspace()
|
if ( ( celestialObjectLocal != null
|
||||||
|
&& celestialObjectLocal.isHyperspace() )
|
||||||
|| celestialObjectRemote.isHyperspace() ) {
|
|| celestialObjectRemote.isHyperspace() ) {
|
||||||
isJammed = true;
|
isJammed = true;
|
||||||
reasonJammed = "Blocked by warp field barrier";
|
reasonJammed = "Blocked by warp field barrier";
|
||||||
|
@ -921,6 +950,9 @@ public class TileEntityTransporterCore extends TileEntityAbstractEnergyConsumer
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// always validate context
|
||||||
|
checkBeaconObsolescence();
|
||||||
|
if (globalPositionBeacon != null) {
|
||||||
if (!isJammed && WarpDriveConfig.LOGGING_TRANSPORTER) {// only log first jamming occurrence
|
if (!isJammed && WarpDriveConfig.LOGGING_TRANSPORTER) {// only log first jamming occurrence
|
||||||
WarpDrive.logger.info(String.format("%s Conflicting beacon requests received %s is not %s",
|
WarpDrive.logger.info(String.format("%s Conflicting beacon requests received %s is not %s",
|
||||||
this, tileEntity, uuid));
|
this, tileEntity, uuid));
|
||||||
|
@ -930,6 +962,7 @@ public class TileEntityTransporterCore extends TileEntityAbstractEnergyConsumer
|
||||||
tickCooldown = Math.max(tickCooldown, WarpDriveConfig.TRANSPORTER_JAMMED_COOLDOWN_TICKS);
|
tickCooldown = Math.max(tickCooldown, WarpDriveConfig.TRANSPORTER_JAMMED_COOLDOWN_TICKS);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// check for new beacon
|
// check for new beacon
|
||||||
if (globalPositionBeacon == null) {
|
if (globalPositionBeacon == null) {
|
||||||
|
@ -938,6 +971,7 @@ public class TileEntityTransporterCore extends TileEntityAbstractEnergyConsumer
|
||||||
isJammed = true;
|
isJammed = true;
|
||||||
reasonJammed = "Beacon request received";
|
reasonJammed = "Beacon request received";
|
||||||
lockStrengthActual = 0.0D;
|
lockStrengthActual = 0.0D;
|
||||||
|
forceChunks();
|
||||||
if (transporterState == EnumTransporterState.ENERGIZING) {
|
if (transporterState == EnumTransporterState.ENERGIZING) {
|
||||||
transporterState = EnumTransporterState.ACQUIRING;
|
transporterState = EnumTransporterState.ACQUIRING;
|
||||||
}
|
}
|
||||||
|
@ -947,6 +981,60 @@ public class TileEntityTransporterCore extends TileEntityAbstractEnergyConsumer
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void forceChunks() {
|
||||||
|
if (ticketChunkloading != null) {
|
||||||
|
WarpDrive.logger.warn(String.format("%s Already chunkloading...",
|
||||||
|
this));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (WarpDriveConfig.LOGGING_TRANSPORTER) {
|
||||||
|
WarpDrive.logger.info(String.format("%s Forcing chunks",
|
||||||
|
this));
|
||||||
|
}
|
||||||
|
ticketChunkloading = ForgeChunkManager.requestTicket(WarpDrive.instance, world, Type.NORMAL);
|
||||||
|
if (ticketChunkloading == null) {
|
||||||
|
WarpDrive.logger.error(String.format("%s Chunkloading rejected",
|
||||||
|
this));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
final AxisAlignedBB aabbArea = getStarMapArea();
|
||||||
|
final int minX = (int) aabbArea.minX >> 4;
|
||||||
|
final int maxX = (int) aabbArea.maxX >> 4;
|
||||||
|
final int minZ = (int) aabbArea.minZ >> 4;
|
||||||
|
final int maxZ = (int) aabbArea.maxZ >> 4;
|
||||||
|
int chunkCount = 0;
|
||||||
|
for (int x = minX; x <= maxX; x++) {
|
||||||
|
for (int z = minZ; z <= maxZ; z++) {
|
||||||
|
chunkCount++;
|
||||||
|
if (chunkCount > ticketChunkloading.getMaxChunkListDepth()) {
|
||||||
|
WarpDrive.logger.error(String.format("%s Too many chunks to load: %d > %d",
|
||||||
|
this,
|
||||||
|
(maxX - minX + 1) * (maxZ - minZ + 1),
|
||||||
|
ticketChunkloading.getMaxChunkListDepth() ));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
ForgeChunkManager.forceChunk(ticketChunkloading, new ChunkPos(x, z));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void releaseChunks() {
|
||||||
|
if (ticketChunkloading == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (WarpDriveConfig.LOGGING_TRANSPORTER) {
|
||||||
|
WarpDrive.logger.info(this + " Releasing chunks");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ticketChunkloading != null) {
|
||||||
|
ForgeChunkManager.releaseTicket(ticketChunkloading);
|
||||||
|
ticketChunkloading = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private static FocusValues getFocusValueAtCoordinates(final World world, final VectorI vLocation, final int radius) {
|
private static FocusValues getFocusValueAtCoordinates(final World world, final VectorI vLocation, final int radius) {
|
||||||
// return default values if world isn't loaded
|
// return default values if world isn't loaded
|
||||||
if (world == null) {
|
if (world == null) {
|
||||||
|
|
Loading…
Reference in a new issue