package gregtech.api.util;
import static gregtech.GT_Mod.GT_FML_LOGGER;
import static gregtech.api.enums.GT_Values.D1;
import static gregtech.api.enums.GT_Values.E;
import static gregtech.api.enums.GT_Values.GT;
import static gregtech.api.enums.GT_Values.L;
import static gregtech.api.enums.GT_Values.M;
import static gregtech.api.enums.GT_Values.NW;
import static gregtech.api.enums.GT_Values.V;
import static gregtech.api.enums.GT_Values.W;
import static gregtech.common.GT_UndergroundOil.undergroundOilReadInformation;
import cofh.api.transport.IItemDuct;
import com.google.auto.value.AutoValue;
import com.google.common.base.Suppliers;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Maps;
import com.gtnewhorizon.structurelib.alignment.IAlignment;
import com.gtnewhorizon.structurelib.alignment.IAlignmentProvider;
import com.mojang.authlib.GameProfile;
import cpw.mods.fml.common.FMLCommonHandler;
import cpw.mods.fml.common.registry.GameRegistry;
import gregtech.api.GregTech_API;
import gregtech.api.damagesources.GT_DamageSources;
import gregtech.api.damagesources.GT_DamageSources.DamageSourceHotItem;
import gregtech.api.enchants.Enchantment_Radioactivity;
import gregtech.api.enums.*;
import gregtech.api.events.BlockScanningEvent;
import gregtech.api.interfaces.IBlockContainer;
import gregtech.api.interfaces.IDebugableBlock;
import gregtech.api.interfaces.IHasIndexedTexture;
import gregtech.api.interfaces.IProjectileItem;
import gregtech.api.interfaces.ITexture;
import gregtech.api.interfaces.tileentity.IBasicEnergyContainer;
import gregtech.api.interfaces.tileentity.ICoverable;
import gregtech.api.interfaces.tileentity.IGregTechDeviceInformation;
import gregtech.api.interfaces.tileentity.IGregTechTileEntity;
import gregtech.api.interfaces.tileentity.IMachineProgress;
import gregtech.api.interfaces.tileentity.IUpgradableMachine;
import gregtech.api.items.GT_EnergyArmor_Item;
import gregtech.api.items.GT_Generic_Item;
import gregtech.api.items.GT_MetaGenerated_Tool;
import gregtech.api.net.GT_Packet_Sound;
import gregtech.api.objects.CollectorUtils;
import gregtech.api.objects.GT_ItemStack;
import gregtech.api.objects.GT_ItemStack2;
import gregtech.api.objects.ItemData;
import gregtech.api.threads.GT_Runnable_Sound;
import gregtech.api.util.extensions.ArrayExt;
import gregtech.common.GT_Pollution;
import gregtech.common.blocks.GT_Block_Ores_Abstract;
import ic2.api.recipe.IRecipeInput;
import ic2.api.recipe.RecipeInputItemStack;
import ic2.api.recipe.RecipeInputOreDict;
import ic2.api.recipe.RecipeOutput;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.math.BigInteger;
import java.math.RoundingMode;
import java.text.DecimalFormat;
import java.text.DecimalFormatSymbols;
import java.util.*;
import java.util.Map.Entry;
import java.util.function.Function;
import java.util.function.IntFunction;
import java.util.function.Supplier;
import javax.annotation.Nullable;
import net.minecraft.block.Block;
import net.minecraft.enchantment.Enchantment;
import net.minecraft.enchantment.EnchantmentHelper;
import net.minecraft.entity.Entity;
import net.minecraft.entity.EntityList;
import net.minecraft.entity.EntityLiving;
import net.minecraft.entity.EntityLivingBase;
import net.minecraft.entity.EnumCreatureAttribute;
import net.minecraft.entity.item.EntityItem;
import net.minecraft.entity.player.EntityPlayer;
import net.minecraft.entity.player.EntityPlayerMP;
import net.minecraft.init.Blocks;
import net.minecraft.init.Items;
import net.minecraft.inventory.IInventory;
import net.minecraft.inventory.ISidedInventory;
import net.minecraft.item.Item;
import net.minecraft.item.ItemStack;
import net.minecraft.nbt.NBTBase;
import net.minecraft.nbt.NBTBase.NBTPrimitive;
import net.minecraft.nbt.NBTTagCompound;
import net.minecraft.nbt.NBTTagList;
import net.minecraft.nbt.NBTTagString;
import net.minecraft.network.play.server.S07PacketRespawn;
import net.minecraft.network.play.server.S1DPacketEntityEffect;
import net.minecraft.network.play.server.S1FPacketSetExperience;
import net.minecraft.potion.Potion;
import net.minecraft.potion.PotionEffect;
import net.minecraft.tileentity.TileEntity;
import net.minecraft.tileentity.TileEntityChest;
import net.minecraft.util.*;
import net.minecraft.world.World;
import net.minecraft.world.WorldServer;
import net.minecraft.world.chunk.Chunk;
import net.minecraftforge.common.DimensionManager;
import net.minecraftforge.common.MinecraftForge;
import net.minecraftforge.common.util.BlockSnapshot;
import net.minecraftforge.common.util.FakePlayer;
import net.minecraftforge.common.util.FakePlayerFactory;
import net.minecraftforge.common.util.ForgeDirection;
import net.minecraftforge.event.ForgeEventFactory;
import net.minecraftforge.event.world.BlockEvent;
import net.minecraftforge.fluids.Fluid;
import net.minecraftforge.fluids.FluidContainerRegistry.FluidContainerData;
import net.minecraftforge.fluids.FluidRegistry;
import net.minecraftforge.fluids.FluidStack;
import net.minecraftforge.fluids.FluidTankInfo;
import net.minecraftforge.fluids.IFluidContainerItem;
import net.minecraftforge.fluids.IFluidHandler;
import net.minecraftforge.oredict.OreDictionary;
/**
* NEVER INCLUDE THIS FILE IN YOUR MOD!!!
*
* Just a few Utility Functions I use.
*/
public class GT_Utility {
/** Formats a number with group separator and at most 2 fraction digits. */
private static final Map decimalFormatters = new HashMap<>();
/**
* Forge screwed the Fluid Registry up again, so I make my own, which is also much more efficient than the stupid Stuff over there.
*/
private static final List sFluidContainerList = new ArrayList<>();
private static final Map sFilledContainerToData = new /*Concurrent*/ HashMap<>();
private static final Map> sEmptyContainerToFluidToData =
new /*Concurrent*/ HashMap<>();
private static final Map> sFluidToContainers = new HashMap<>();
private static final Map sFluidUnlocalizedNameToFluid = new HashMap<>();
/** Must use {@code Supplier} here because the ore prefixes have not yet been registered at class load time. */
private static final Map> sOreToCobble = new HashMap<>();
private static final Map sOreTable = new HashMap<>();
public static volatile int VERSION = 509;
public static boolean TE_CHECK = false, BC_CHECK = false, CHECK_ALL = true, RF_CHECK = false;
public static Map sPlayedSoundMap = new /*Concurrent*/ HashMap<>();
private static int sBookCount = 0;
public static UUID defaultUuid =
null; // maybe default non-null? UUID.fromString("00000000-0000-0000-0000-000000000000");
static {
GregTech_API.sItemStackMappings.add(sFilledContainerToData);
GregTech_API.sItemStackMappings.add(sEmptyContainerToFluidToData);
// 1 is the magic index to get the cobblestone block.
// See: GT_Block_Stones.java, GT_Block_Granites.java
Function> materialToCobble = m -> Suppliers.memoize(
() -> GT_OreDictUnificator.getOres(OrePrefixes.stone, m).get(1))::get;
sOreToCobble.put(OrePrefixes.oreBlackgranite, materialToCobble.apply(Materials.GraniteBlack));
sOreToCobble.put(OrePrefixes.oreRedgranite, materialToCobble.apply(Materials.GraniteRed));
sOreToCobble.put(OrePrefixes.oreMarble, materialToCobble.apply(Materials.Marble));
sOreToCobble.put(OrePrefixes.oreBasalt, materialToCobble.apply(Materials.Basalt));
sOreToCobble.put(OrePrefixes.oreNetherrack, () -> new ItemStack(Blocks.netherrack));
sOreToCobble.put(OrePrefixes.oreEndstone, () -> new ItemStack(Blocks.end_stone));
}
public static int safeInt(long number, int margin) {
return number > Integer.MAX_VALUE - margin ? Integer.MAX_VALUE - margin : (int) number;
}
public static int safeInt(long number) {
return number > V[V.length - 1]
? safeInt(V[V.length - 1], 1)
: number < Integer.MIN_VALUE ? Integer.MIN_VALUE : (int) number;
}
public static Field getPublicField(Object aObject, String aField) {
Field rField = null;
try {
rField = aObject.getClass().getDeclaredField(aField);
} catch (Throwable e) {
/*Do nothing*/
}
return rField;
}
public static Field getField(Object aObject, String aField) {
Field rField = null;
try {
rField = aObject.getClass().getDeclaredField(aField);
rField.setAccessible(true);
} catch (Throwable e) {
/*Do nothing*/
}
return rField;
}
public static Field getField(Class aObject, String aField) {
Field rField = null;
try {
rField = aObject.getDeclaredField(aField);
rField.setAccessible(true);
} catch (Throwable e) {
/*Do nothing*/
}
return rField;
}
public static Method getMethod(Class aObject, String aMethod, Class>... aParameterTypes) {
Method rMethod = null;
try {
rMethod = aObject.getMethod(aMethod, aParameterTypes);
rMethod.setAccessible(true);
} catch (Throwable e) {
/*Do nothing*/
}
return rMethod;
}
public static Method getMethod(Object aObject, String aMethod, Class>... aParameterTypes) {
Method rMethod = null;
try {
rMethod = aObject.getClass().getMethod(aMethod, aParameterTypes);
rMethod.setAccessible(true);
} catch (Throwable e) {
/*Do nothing*/
}
return rMethod;
}
public static Field getField(Object aObject, String aField, boolean aPrivate, boolean aLogErrors) {
try {
Field tField = (aObject instanceof Class)
? ((Class) aObject).getDeclaredField(aField)
: (aObject instanceof String)
? Class.forName((String) aObject).getDeclaredField(aField)
: aObject.getClass().getDeclaredField(aField);
if (aPrivate) tField.setAccessible(true);
return tField;
} catch (Throwable e) {
if (aLogErrors) e.printStackTrace(GT_Log.err);
}
return null;
}
public static Object getFieldContent(Object aObject, String aField, boolean aPrivate, boolean aLogErrors) {
try {
Field tField = (aObject instanceof Class)
? ((Class) aObject).getDeclaredField(aField)
: (aObject instanceof String)
? Class.forName((String) aObject).getDeclaredField(aField)
: aObject.getClass().getDeclaredField(aField);
if (aPrivate) tField.setAccessible(true);
return tField.get(aObject instanceof Class || aObject instanceof String ? null : aObject);
} catch (Throwable e) {
if (aLogErrors) e.printStackTrace(GT_Log.err);
}
return null;
}
public static Object callPublicMethod(Object aObject, String aMethod, Object... aParameters) {
return callMethod(aObject, aMethod, false, false, true, aParameters);
}
public static Object callPrivateMethod(Object aObject, String aMethod, Object... aParameters) {
return callMethod(aObject, aMethod, true, false, true, aParameters);
}
public static Object callMethod(
Object aObject,
String aMethod,
boolean aPrivate,
boolean aUseUpperCasedDataTypes,
boolean aLogErrors,
Object... aParameters) {
try {
Class>[] tParameterTypes = new Class>[aParameters.length];
for (byte i = 0; i < aParameters.length; i++) {
if (aParameters[i] instanceof Class) {
tParameterTypes[i] = (Class) aParameters[i];
aParameters[i] = null;
} else {
tParameterTypes[i] = aParameters[i].getClass();
}
if (!aUseUpperCasedDataTypes) {
if (tParameterTypes[i] == Boolean.class) tParameterTypes[i] = boolean.class;
else if (tParameterTypes[i] == Byte.class) tParameterTypes[i] = byte.class;
else if (tParameterTypes[i] == Short.class) tParameterTypes[i] = short.class;
else if (tParameterTypes[i] == Integer.class) tParameterTypes[i] = int.class;
else if (tParameterTypes[i] == Long.class) tParameterTypes[i] = long.class;
else if (tParameterTypes[i] == Float.class) tParameterTypes[i] = float.class;
else if (tParameterTypes[i] == Double.class) tParameterTypes[i] = double.class;
}
}
Method tMethod = (aObject instanceof Class)
? ((Class) aObject).getMethod(aMethod, tParameterTypes)
: aObject.getClass().getMethod(aMethod, tParameterTypes);
if (aPrivate) tMethod.setAccessible(true);
return tMethod.invoke(aObject, aParameters);
} catch (Throwable e) {
if (aLogErrors) e.printStackTrace(GT_Log.err);
}
return null;
}
public static Object callConstructor(
String aClass,
int aConstructorIndex,
Object aReplacementObject,
boolean aLogErrors,
Object... aParameters) {
try {
return callConstructor(
Class.forName(aClass), aConstructorIndex, aReplacementObject, aLogErrors, aParameters);
} catch (Throwable e) {
if (aLogErrors) e.printStackTrace(GT_Log.err);
}
return aReplacementObject;
}
public static Object callConstructor(
Class> aClass,
int aConstructorIndex,
Object aReplacementObject,
boolean aLogErrors,
Object... aParameters) {
if (aConstructorIndex < 0) {
try {
for (Constructor> tConstructor : aClass.getConstructors()) {
try {
return tConstructor.newInstance(aParameters);
} catch (Throwable ignored) {
}
}
} catch (Throwable e) {
if (aLogErrors) e.printStackTrace(GT_Log.err);
}
} else {
try {
return aClass.getConstructors()[aConstructorIndex].newInstance(aParameters);
} catch (Throwable e) {
if (aLogErrors) e.printStackTrace(GT_Log.err);
}
}
return aReplacementObject;
}
public static String capitalizeString(String aString) {
if (aString != null && aString.length() > 0)
return aString.substring(0, 1).toUpperCase() + aString.substring(1);
return E;
}
public static boolean getPotion(EntityLivingBase aPlayer, int aPotionIndex) {
try {
Field tPotionHashmap = null;
Field[] fields = EntityLiving.class.getDeclaredFields();
for (Field field : fields) {
if (field.getType() == HashMap.class) {
tPotionHashmap = field;
tPotionHashmap.setAccessible(true);
break;
}
}
if (tPotionHashmap != null) return ((HashMap) tPotionHashmap.get(aPlayer)).get(aPotionIndex) != null;
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
return false;
}
public static String getClassName(Object aObject) {
if (aObject == null) return "null";
return aObject.getClass()
.getName()
.substring(aObject.getClass().getName().lastIndexOf(".") + 1);
}
public static void removePotion(EntityLivingBase aPlayer, int aPotionIndex) {
try {
Field tPotionHashmap = null;
Field[] fields = EntityLiving.class.getDeclaredFields();
for (Field field : fields) {
if (field.getType() == HashMap.class) {
tPotionHashmap = field;
tPotionHashmap.setAccessible(true);
break;
}
}
if (tPotionHashmap != null) ((HashMap) tPotionHashmap.get(aPlayer)).remove(aPotionIndex);
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
}
public static boolean getFullInvisibility(EntityPlayer aPlayer) {
try {
if (aPlayer.isInvisible()) {
for (int i = 0; i < 4; i++) {
if (aPlayer.inventory.armorInventory[i] != null) {
if (aPlayer.inventory.armorInventory[i].getItem() instanceof GT_EnergyArmor_Item) {
if ((((GT_EnergyArmor_Item) aPlayer.inventory.armorInventory[i].getItem()).mSpecials & 512)
!= 0) {
if (GT_ModHandler.canUseElectricItem(aPlayer.inventory.armorInventory[i], 10000)) {
return true;
}
}
}
}
}
}
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
return false;
}
public static ItemStack suckOneItemStackAt(
World aWorld, double aX, double aY, double aZ, double aL, double aH, double aW) {
for (EntityItem tItem : (ArrayList) aWorld.getEntitiesWithinAABB(
EntityItem.class, AxisAlignedBB.getBoundingBox(aX, aY, aZ, aX + aL, aY + aH, aZ + aW))) {
if (!tItem.isDead) {
aWorld.removeEntity(tItem);
tItem.setDead();
return tItem.getEntityItem();
}
}
return null;
}
public static byte getOppositeSide(int aSide) {
return (byte) ForgeDirection.getOrientation(aSide).getOpposite().ordinal();
}
public static byte getTier(long l) {
byte i = -1;
while (++i < V.length) if (l <= V[i]) return i;
return i;
}
public static void sendChatToPlayer(EntityPlayer aPlayer, String aChatMessage) {
if (aPlayer instanceof EntityPlayerMP && aChatMessage != null) {
aPlayer.addChatComponentMessage(new ChatComponentText(aChatMessage));
}
}
public static void checkAvailabilities() {
if (CHECK_ALL) {
try {
Class tClass = IItemDuct.class;
tClass.getCanonicalName();
TE_CHECK = true;
} catch (Throwable e) {
/**/
}
try {
Class tClass = buildcraft.api.transport.IPipeTile.class;
tClass.getCanonicalName();
BC_CHECK = true;
} catch (Throwable e) {
/**/
}
try {
Class tClass = cofh.api.energy.IEnergyReceiver.class;
tClass.getCanonicalName();
RF_CHECK = true;
} catch (Throwable e) {
/**/
}
CHECK_ALL = false;
}
}
public static boolean isConnectableNonInventoryPipe(Object aTileEntity, int aSide) {
if (aTileEntity == null) return false;
checkAvailabilities();
if (TE_CHECK && aTileEntity instanceof IItemDuct) return true;
if (BC_CHECK && aTileEntity instanceof buildcraft.api.transport.IPipeTile)
return ((buildcraft.api.transport.IPipeTile) aTileEntity)
.isPipeConnected(ForgeDirection.getOrientation(aSide));
return GregTech_API.mTranslocator && aTileEntity instanceof codechicken.translocator.TileItemTranslocator;
}
/**
* Moves Stack from Inv-Slot to Inv-Slot, without checking if its even allowed.
*
* @return the Amount of moved Items
*/
public static byte moveStackIntoPipe(
IInventory aTileEntity1,
Object aTileEntity2,
int[] aGrabSlots,
int aGrabFrom,
int aPutTo,
List aFilter,
boolean aInvertFilter,
byte aMaxTargetStackSize,
byte aMinTargetStackSize,
byte aMaxMoveAtOnce,
byte aMinMoveAtOnce) {
return moveStackIntoPipe(
aTileEntity1,
aTileEntity2,
aGrabSlots,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
true);
}
/**
* Moves Stack from Inv-Slot to Inv-Slot, without checking if its even allowed.
*
* @return the Amount of moved Items
*/
public static byte moveStackIntoPipe(
IInventory aTileEntity1,
Object aTileEntity2,
int[] aGrabSlots,
int aGrabFrom,
int aPutTo,
List aFilter,
boolean aInvertFilter,
byte aMaxTargetStackSize,
byte aMinTargetStackSize,
byte aMaxMoveAtOnce,
byte aMinMoveAtOnce,
boolean dropItem) {
if (aTileEntity1 == null
|| aMaxTargetStackSize <= 0
|| aMinTargetStackSize <= 0
|| aMinTargetStackSize > aMaxTargetStackSize
|| aMaxMoveAtOnce <= 0
|| aMinMoveAtOnce > aMaxMoveAtOnce) return 0;
if (aTileEntity2 != null) {
checkAvailabilities();
if (TE_CHECK && aTileEntity2 instanceof IItemDuct) {
for (int aGrabSlot : aGrabSlots) {
if (listContainsItem(aFilter, aTileEntity1.getStackInSlot(aGrabSlot), true, aInvertFilter)) {
if (isAllowedToTakeFromSlot(
aTileEntity1, aGrabSlot, (byte) aGrabFrom, aTileEntity1.getStackInSlot(aGrabSlot))) {
if (Math.max(aMinMoveAtOnce, aMinTargetStackSize)
<= aTileEntity1.getStackInSlot(aGrabSlot).stackSize) {
ItemStack tStack = copyAmount(
Math.min(
aTileEntity1.getStackInSlot(aGrabSlot).stackSize,
Math.min(aMaxMoveAtOnce, aMaxTargetStackSize)),
aTileEntity1.getStackInSlot(aGrabSlot));
ItemStack rStack = ((IItemDuct) aTileEntity2)
.insertItem(ForgeDirection.getOrientation(aPutTo), copyOrNull(tStack));
byte tMovedItemCount =
(byte) (tStack.stackSize - (rStack == null ? 0 : rStack.stackSize));
if (tMovedItemCount >= 1 /*Math.max(aMinMoveAtOnce, aMinTargetStackSize)*/) {
// ((cofh.api.transport.IItemConduit)aTileEntity2).insertItem(ForgeDirection.getOrientation(aPutTo), copyAmount(tMovedItemCount, tStack), F);
aTileEntity1.decrStackSize(aGrabSlot, tMovedItemCount);
aTileEntity1.markDirty();
return tMovedItemCount;
}
}
}
}
}
return 0;
}
if (BC_CHECK && aTileEntity2 instanceof buildcraft.api.transport.IPipeTile) {
for (int aGrabSlot : aGrabSlots) {
if (listContainsItem(aFilter, aTileEntity1.getStackInSlot(aGrabSlot), true, aInvertFilter)) {
if (isAllowedToTakeFromSlot(
aTileEntity1, aGrabSlot, (byte) aGrabFrom, aTileEntity1.getStackInSlot(aGrabSlot))) {
if (Math.max(aMinMoveAtOnce, aMinTargetStackSize)
<= aTileEntity1.getStackInSlot(aGrabSlot).stackSize) {
ItemStack tStack = copyAmount(
Math.min(
aTileEntity1.getStackInSlot(aGrabSlot).stackSize,
Math.min(aMaxMoveAtOnce, aMaxTargetStackSize)),
aTileEntity1.getStackInSlot(aGrabSlot));
byte tMovedItemCount = (byte) ((buildcraft.api.transport.IPipeTile) aTileEntity2)
.injectItem(copyOrNull(tStack), false, ForgeDirection.getOrientation(aPutTo));
if (tMovedItemCount >= Math.max(aMinMoveAtOnce, aMinTargetStackSize)) {
tMovedItemCount = (byte) (((buildcraft.api.transport.IPipeTile) aTileEntity2)
.injectItem(
copyAmount(tMovedItemCount, tStack),
true,
ForgeDirection.getOrientation(aPutTo)));
aTileEntity1.decrStackSize(aGrabSlot, tMovedItemCount);
aTileEntity1.markDirty();
return tMovedItemCount;
}
}
}
}
}
return 0;
}
}
ForgeDirection tDirection = ForgeDirection.getOrientation(aGrabFrom);
if (aTileEntity1 instanceof TileEntity
&& tDirection != ForgeDirection.UNKNOWN
&& tDirection.getOpposite() == ForgeDirection.getOrientation(aPutTo)) {
int tX = ((TileEntity) aTileEntity1).xCoord + tDirection.offsetX,
tY = ((TileEntity) aTileEntity1).yCoord + tDirection.offsetY,
tZ = ((TileEntity) aTileEntity1).zCoord + tDirection.offsetZ;
if (!hasBlockHitBox(((TileEntity) aTileEntity1).getWorldObj(), tX, tY, tZ) && dropItem) {
for (int aGrabSlot : aGrabSlots) {
if (listContainsItem(aFilter, aTileEntity1.getStackInSlot(aGrabSlot), true, aInvertFilter)) {
if (isAllowedToTakeFromSlot(
aTileEntity1, aGrabSlot, (byte) aGrabFrom, aTileEntity1.getStackInSlot(aGrabSlot))) {
if (Math.max(aMinMoveAtOnce, aMinTargetStackSize)
<= aTileEntity1.getStackInSlot(aGrabSlot).stackSize) {
ItemStack tStack = copyAmount(
Math.min(
aTileEntity1.getStackInSlot(aGrabSlot).stackSize,
Math.min(aMaxMoveAtOnce, aMaxTargetStackSize)),
aTileEntity1.getStackInSlot(aGrabSlot));
EntityItem tEntity = new EntityItem(
((TileEntity) aTileEntity1).getWorldObj(),
tX + 0.5,
tY + 0.5,
tZ + 0.5,
tStack);
tEntity.motionX = tEntity.motionY = tEntity.motionZ = 0;
((TileEntity) aTileEntity1).getWorldObj().spawnEntityInWorld(tEntity);
aTileEntity1.decrStackSize(aGrabSlot, tStack.stackSize);
aTileEntity1.markDirty();
return (byte) tStack.stackSize;
}
}
}
}
}
}
return 0;
}
/**
* Moves Stack from Inv-Slot to Inv-Slot, without checking if its even allowed. (useful for internal Inventory Operations)
*
* @return the Amount of moved Items
*/
public static byte moveStackFromSlotAToSlotB(
IInventory aTileEntity1,
IInventory aTileEntity2,
int aGrabFrom,
int aPutTo,
byte aMaxTargetStackSize,
byte aMinTargetStackSize,
byte aMaxMoveAtOnce,
byte aMinMoveAtOnce) {
if (aTileEntity1 == null
|| aTileEntity2 == null
|| aMaxTargetStackSize <= 0
|| aMinTargetStackSize <= 0
|| aMinTargetStackSize > aMaxTargetStackSize
|| aMaxMoveAtOnce <= 0
|| aMinMoveAtOnce > aMaxMoveAtOnce) return 0;
ItemStack tStack1 = aTileEntity1.getStackInSlot(aGrabFrom),
tStack2 = aTileEntity2.getStackInSlot(aPutTo),
tStack3 = null;
if (tStack1 != null) {
if (tStack2 != null && !areStacksEqual(tStack1, tStack2)) return 0;
tStack3 = copyOrNull(tStack1);
aMaxTargetStackSize = (byte) Math.min(
aMaxTargetStackSize,
Math.min(
tStack3.getMaxStackSize(),
Math.min(
tStack2 == null ? Integer.MAX_VALUE : tStack2.getMaxStackSize(),
aTileEntity2.getInventoryStackLimit())));
tStack3.stackSize =
Math.min(tStack3.stackSize, aMaxTargetStackSize - (tStack2 == null ? 0 : tStack2.stackSize));
if (tStack3.stackSize > aMaxMoveAtOnce) tStack3.stackSize = aMaxMoveAtOnce;
if (tStack3.stackSize + (tStack2 == null ? 0 : tStack2.stackSize)
>= Math.min(tStack3.getMaxStackSize(), aMinTargetStackSize)
&& tStack3.stackSize >= aMinMoveAtOnce) {
tStack3 = aTileEntity1.decrStackSize(aGrabFrom, tStack3.stackSize);
aTileEntity1.markDirty();
if (tStack3 != null) {
if (tStack2 == null) {
aTileEntity2.setInventorySlotContents(aPutTo, copyOrNull(tStack3));
} else {
tStack2.stackSize += tStack3.stackSize;
}
aTileEntity2.markDirty();
return (byte) tStack3.stackSize;
}
}
}
return 0;
}
public static boolean isAllowedToTakeFromSlot(IInventory aTileEntity, int aSlot, byte aSide, ItemStack aStack) {
if (ForgeDirection.getOrientation(aSide) == ForgeDirection.UNKNOWN) {
return isAllowedToTakeFromSlot(aTileEntity, aSlot, (byte) 0, aStack)
|| isAllowedToTakeFromSlot(aTileEntity, aSlot, (byte) 1, aStack)
|| isAllowedToTakeFromSlot(aTileEntity, aSlot, (byte) 2, aStack)
|| isAllowedToTakeFromSlot(aTileEntity, aSlot, (byte) 3, aStack)
|| isAllowedToTakeFromSlot(aTileEntity, aSlot, (byte) 4, aStack)
|| isAllowedToTakeFromSlot(aTileEntity, aSlot, (byte) 5, aStack);
}
if (aTileEntity instanceof ISidedInventory)
return ((ISidedInventory) aTileEntity).canExtractItem(aSlot, aStack, aSide);
return true;
}
public static boolean isAllowedToPutIntoSlot(
IInventory aTileEntity, int aSlot, byte aSide, ItemStack aStack, byte aMaxStackSize) {
ItemStack tStack = aTileEntity.getStackInSlot(aSlot);
if (tStack != null && (!areStacksEqual(tStack, aStack) || tStack.stackSize >= tStack.getMaxStackSize()))
return false;
if (ForgeDirection.getOrientation(aSide) == ForgeDirection.UNKNOWN) {
return isAllowedToPutIntoSlot(aTileEntity, aSlot, (byte) 0, aStack, aMaxStackSize)
|| isAllowedToPutIntoSlot(aTileEntity, aSlot, (byte) 1, aStack, aMaxStackSize)
|| isAllowedToPutIntoSlot(aTileEntity, aSlot, (byte) 2, aStack, aMaxStackSize)
|| isAllowedToPutIntoSlot(aTileEntity, aSlot, (byte) 3, aStack, aMaxStackSize)
|| isAllowedToPutIntoSlot(aTileEntity, aSlot, (byte) 4, aStack, aMaxStackSize)
|| isAllowedToPutIntoSlot(aTileEntity, aSlot, (byte) 5, aStack, aMaxStackSize);
}
if (aTileEntity instanceof ISidedInventory
&& !((ISidedInventory) aTileEntity).canInsertItem(aSlot, aStack, aSide)) return false;
return aSlot < aTileEntity.getSizeInventory() && aTileEntity.isItemValidForSlot(aSlot, aStack);
}
/**
* moves multiple stacks from Inv-Side to Inv-Side
*
* @return the Amount of moved Items
*/
public static int moveMultipleItemStacks(
Object aTileEntity1,
Object aTileEntity2,
byte aGrabFrom,
byte aPutTo,
List aFilter,
boolean aInvertFilter,
byte aMaxTargetStackSize,
byte aMinTargetStackSize,
byte aMaxMoveAtOnce,
byte aMinMoveAtOnce,
int aStackAmount) {
if (aTileEntity1 instanceof IInventory)
return moveMultipleItemStacks(
(IInventory) aTileEntity1,
aTileEntity2,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
aStackAmount,
true);
return 0;
}
public static int moveMultipleItemStacks(
IInventory aTileEntity1,
Object aTileEntity2,
byte aGrabFrom,
byte aPutTo,
List aFilter,
boolean aInvertFilter,
byte aMaxTargetStackSize,
byte aMinTargetStackSize,
byte aMaxMoveAtOnce,
byte aMinMoveAtOnce,
int aMaxStackTransfer,
boolean aDoCheckChests) {
if (aTileEntity1 == null
|| aMaxTargetStackSize <= 0
|| aMinTargetStackSize <= 0
|| aMaxMoveAtOnce <= 0
|| aMinTargetStackSize > aMaxTargetStackSize
|| aMinMoveAtOnce > aMaxMoveAtOnce
|| aMaxStackTransfer == 0) return 0;
// find where to take from
int[] tGrabSlots = new int[aTileEntity1.getSizeInventory()];
int tGrabSlotsSize = 0;
if (aTileEntity1 instanceof ISidedInventory) {
for (int i : ((ISidedInventory) aTileEntity1).getAccessibleSlotsFromSide(aGrabFrom)) {
ItemStack s = aTileEntity1.getStackInSlot(i);
if (s == null
|| !isAllowedToTakeFromSlot(aTileEntity1, i, aGrabFrom, s)
|| s.stackSize < aMinMoveAtOnce
|| !listContainsItem(aFilter, s, true, aInvertFilter)) continue;
tGrabSlots[tGrabSlotsSize++] = i;
}
} else {
for (int i = 0; i < tGrabSlots.length; i++) {
ItemStack s = aTileEntity1.getStackInSlot(i);
if (s == null || s.stackSize < aMinMoveAtOnce || !listContainsItem(aFilter, s, true, aInvertFilter))
continue;
tGrabSlots[tGrabSlotsSize++] = i;
}
}
// no source, bail out
if (tGrabSlotsSize == 0) {
// maybe source is a double chest. check it
if (aDoCheckChests && aTileEntity1 instanceof TileEntityChest)
return moveFromAdjacentChests(
(TileEntityChest) aTileEntity1,
aTileEntity2,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
aMaxStackTransfer);
return 0;
}
// if target is an inventory, e.g. chest, machine, drawers...
if (aTileEntity2 instanceof IInventory) {
IInventory tPutInventory = (IInventory) aTileEntity2;
// partially filled slot spare space mapping.
// value is the sum of all spare space left not counting completely empty slot
HashMap tPutItems = new HashMap<>(tPutInventory.getSizeInventory());
// partially filled slot contents
HashMap> tPutItemStacks = new HashMap<>(tPutInventory.getSizeInventory());
// completely empty slots
List tPutFreeSlots = new ArrayList<>(tPutInventory.getSizeInventory());
// find possible target slots
int[] accessibleSlots = null;
if (aTileEntity2 instanceof ISidedInventory)
accessibleSlots = ((ISidedInventory) tPutInventory).getAccessibleSlotsFromSide(aPutTo);
for (int i = 0; i < tPutInventory.getSizeInventory(); i++) {
int slot = i;
if (accessibleSlots != null) {
if (accessibleSlots.length <= i) break;
slot = accessibleSlots[slot];
}
ItemStack s = tPutInventory.getStackInSlot(slot);
if (s == null) {
tPutFreeSlots.add(slot);
} else if ((s.stackSize < s.getMaxStackSize() && s.stackSize < tPutInventory.getInventoryStackLimit())
&& aMinMoveAtOnce <= s.getMaxStackSize() - s.stackSize) {
ItemId sID = ItemId.createNoCopy(s);
tPutItems.merge(
sID,
(Math.min(s.getMaxStackSize(), tPutInventory.getInventoryStackLimit()) - s.stackSize),
Integer::sum);
tPutItemStacks.computeIfAbsent(sID, k -> new ArrayList<>()).add(s);
}
}
// target completely filled, bail out
if (tPutItems.isEmpty() && tPutFreeSlots.isEmpty()) {
// maybe target is a double chest. check it.
if (aDoCheckChests && aTileEntity2 instanceof TileEntityChest)
return moveToAdjacentChests(
aTileEntity1,
(TileEntityChest) aTileEntity2,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
aMaxStackTransfer);
return 0;
}
// go over source stacks one by one
int tStacksMoved = 0, tTotalItemsMoved = 0;
for (int j = 0; j < tGrabSlotsSize; j++) {
int grabSlot = tGrabSlots[j];
int tMovedItems;
int tStackSize;
do {
tMovedItems = 0;
ItemStack tGrabStack = aTileEntity1.getStackInSlot(grabSlot);
if (tGrabStack == null) break;
tStackSize = tGrabStack.stackSize;
ItemId sID = ItemId.createNoCopy(tGrabStack);
if (tPutItems.containsKey(sID)) {
// there is a partially filled slot, try merging
int canPut = Math.min(tPutItems.get(sID), aMaxMoveAtOnce);
if (canPut >= aMinMoveAtOnce) {
List putStack = tPutItemStacks.get(sID);
if (!putStack.isEmpty()) {
// can move, do merge
int toPut = Math.min(canPut, tStackSize);
tMovedItems = toPut;
for (int i = 0; i < putStack.size(); i++) {
ItemStack s = putStack.get(i);
int sToPut = Math.min(
Math.min(
Math.min(toPut, s.getMaxStackSize() - s.stackSize),
tPutInventory.getInventoryStackLimit() - s.stackSize),
aMaxTargetStackSize - s.stackSize);
if (sToPut <= 0) continue;
if (sToPut < aMinMoveAtOnce) continue;
if (s.stackSize + sToPut < aMinTargetStackSize) continue;
toPut -= sToPut;
s.stackSize += sToPut;
if (s.stackSize == s.getMaxStackSize()
|| s.stackSize == tPutInventory.getInventoryStackLimit()) {
// this slot is full. remove this stack from candidate list
putStack.remove(i);
i--;
}
if (toPut == 0) break;
}
tMovedItems -= toPut;
if (tMovedItems > 0) {
tStackSize -= tMovedItems;
tTotalItemsMoved += tMovedItems;
// deduct spare space
tPutItems.merge(sID, tMovedItems, (a, b) -> a.equals(b) ? null : a - b);
if (tStackSize == 0) aTileEntity1.setInventorySlotContents(grabSlot, null);
else tGrabStack.stackSize = tStackSize;
aTileEntity1.markDirty();
tPutInventory.markDirty();
}
}
}
}
// still stuff to move & have completely empty slots
if (tStackSize > 0 && !tPutFreeSlots.isEmpty()) {
for (int i = 0; i < tPutFreeSlots.size(); i++) {
int tPutSlot = tPutFreeSlots.get(i);
if (isAllowedToPutIntoSlot(tPutInventory, tPutSlot, aPutTo, tGrabStack, (byte) 64)) {
// allowed, now do moving
int tMoved = moveStackFromSlotAToSlotB(
aTileEntity1,
tPutInventory,
grabSlot,
tPutSlot,
aMaxTargetStackSize,
aMinTargetStackSize,
(byte) (aMaxMoveAtOnce - tMovedItems),
aMinMoveAtOnce);
if (tMoved > 0) {
ItemStack s = tPutInventory.getStackInSlot(tPutSlot);
if (s != null) {
// s might be null if tPutInventory is very special, e.g. infinity chest
// if s is null, we will not mark this slot as target candidate for anything
int spare =
Math.min(s.getMaxStackSize(), tPutInventory.getInventoryStackLimit())
- s.stackSize;
if (spare > 0) {
ItemId ssID = ItemId.createNoCopy(s);
// add back to spare space count
tPutItems.merge(ssID, spare, Integer::sum);
// add to partially filled slot list
tPutItemStacks
.computeIfAbsent(ssID, k -> new ArrayList<>())
.add(s);
}
// this is no longer free
tPutFreeSlots.remove(i);
i--;
}
// else -> noop
// this is still a free slot. no need to do anything.
tTotalItemsMoved += tMoved;
tMovedItems += tMoved;
tStackSize -= tMoved;
if (tStackSize == 0) break;
}
}
}
}
if (tMovedItems > 0) {
// check if we have moved enough stacks
if (++tStacksMoved >= aMaxStackTransfer) return tTotalItemsMoved;
}
} while (tMovedItems > 0
&& tStackSize > 0); // support inventories that store more than a stack in a slot
}
// check if source is a double chest, if yes, try move from the adjacent as well
if (aDoCheckChests && aTileEntity1 instanceof TileEntityChest) {
int tAmount = moveFromAdjacentChests(
(TileEntityChest) aTileEntity1,
aTileEntity2,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
aMaxStackTransfer - tStacksMoved);
if (tAmount != 0) return tAmount + tTotalItemsMoved;
}
// check if target is a double chest, if yes, try move to the adjacent as well
if (aDoCheckChests && aTileEntity2 instanceof TileEntityChest) {
int tAmount = moveToAdjacentChests(
aTileEntity1,
(TileEntityChest) aTileEntity2,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
aMaxStackTransfer - tStacksMoved);
if (tAmount != 0) return tAmount + tTotalItemsMoved;
}
return tTotalItemsMoved;
}
// there should be a function to transfer more than 1 stack in a pipe
// however I do not see any ways to improve it. too much work for what it is worth
int tTotalItemsMoved = 0;
int tGrabInventorySize = tGrabSlots.length;
for (int i = 0; i < tGrabInventorySize; i++) {
int tMoved = moveStackIntoPipe(
aTileEntity1,
aTileEntity2,
tGrabSlots,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
aDoCheckChests);
if (tMoved == 0) return tTotalItemsMoved;
else tTotalItemsMoved += tMoved;
}
return 0;
}
private static int moveToAdjacentChests(
IInventory aTileEntity1,
TileEntityChest aTargetChest,
byte aGrabFrom,
byte aPutTo,
List aFilter,
boolean aInvertFilter,
byte aMaxTargetStackSize,
byte aMinTargetStackSize,
byte aMaxMoveAtOnce,
byte aMinMoveAtOnce,
int aMaxStackTransfer) {
if (aTargetChest.adjacentChestChecked) {
if (aTargetChest.adjacentChestXNeg != null) {
return moveMultipleItemStacks(
aTileEntity1,
aTargetChest.adjacentChestXNeg,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
aMaxStackTransfer,
false);
} else if (aTargetChest.adjacentChestZNeg != null) {
return moveMultipleItemStacks(
aTileEntity1,
aTargetChest.adjacentChestZNeg,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
aMaxStackTransfer,
false);
} else if (aTargetChest.adjacentChestXPos != null) {
return moveMultipleItemStacks(
aTileEntity1,
aTargetChest.adjacentChestXPos,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
aMaxStackTransfer,
false);
} else if (aTargetChest.adjacentChestZPos != null) {
return moveMultipleItemStacks(
aTileEntity1,
aTargetChest.adjacentChestZPos,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
aMaxStackTransfer,
false);
}
}
return 0;
}
private static int moveFromAdjacentChests(
TileEntityChest aChest,
Object aTileEntity2,
byte aGrabFrom,
byte aPutTo,
List aFilter,
boolean aInvertFilter,
byte aMaxTargetStackSize,
byte aMinTargetStackSize,
byte aMaxMoveAtOnce,
byte aMinMoveAtOnce,
int aMaxStackTransfer) {
if (aChest.adjacentChestXNeg != null) {
return moveMultipleItemStacks(
aChest.adjacentChestXNeg,
aTileEntity2,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
aMaxStackTransfer,
false);
} else if (aChest.adjacentChestZNeg != null) {
return moveMultipleItemStacks(
aChest.adjacentChestZNeg,
aTileEntity2,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
aMaxStackTransfer,
false);
} else if (aChest.adjacentChestXPos != null) {
return moveMultipleItemStacks(
aChest.adjacentChestXPos,
aTileEntity2,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
aMaxStackTransfer,
false);
} else if (aChest.adjacentChestZPos != null) {
return moveMultipleItemStacks(
aChest.adjacentChestZPos,
aTileEntity2,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
aMaxStackTransfer,
false);
}
return 0;
}
/**
* Moves Stack from Inv-Side to Inv-Side.
*
* @return the Amount of moved Items
*/
public static byte moveOneItemStack(
Object aTileEntity1,
Object aTileEntity2,
byte aGrabFrom,
byte aPutTo,
List aFilter,
boolean aInvertFilter,
byte aMaxTargetStackSize,
byte aMinTargetStackSize,
byte aMaxMoveAtOnce,
byte aMinMoveAtOnce) {
if (aTileEntity1 instanceof IInventory)
return moveOneItemStack(
(IInventory) aTileEntity1,
aTileEntity2,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
true);
return 0;
}
/**
* This is only because I needed an additional Parameter for the Double Chest Check.
*/
private static byte moveOneItemStack(
IInventory aTileEntity1,
Object aTileEntity2,
byte aGrabFrom,
byte aPutTo,
List aFilter,
boolean aInvertFilter,
byte aMaxTargetStackSize,
byte aMinTargetStackSize,
byte aMaxMoveAtOnce,
byte aMinMoveAtOnce,
boolean aDoCheckChests) {
if (aTileEntity1 == null
|| aMaxTargetStackSize <= 0
|| aMinTargetStackSize <= 0
|| aMaxMoveAtOnce <= 0
|| aMinTargetStackSize > aMaxTargetStackSize
|| aMinMoveAtOnce > aMaxMoveAtOnce) return 0;
int[] tGrabSlots = null;
if (aTileEntity1 instanceof ISidedInventory)
tGrabSlots = ((ISidedInventory) aTileEntity1).getAccessibleSlotsFromSide(aGrabFrom);
if (tGrabSlots == null) {
tGrabSlots = new int[aTileEntity1.getSizeInventory()];
for (int i = 0; i < tGrabSlots.length; i++) tGrabSlots[i] = i;
}
if (aTileEntity2 instanceof IInventory) {
int[] tPutSlots = null;
if (aTileEntity2 instanceof ISidedInventory)
tPutSlots = ((ISidedInventory) aTileEntity2).getAccessibleSlotsFromSide(aPutTo);
if (tPutSlots == null) {
tPutSlots = new int[((IInventory) aTileEntity2).getSizeInventory()];
for (int i = 0; i < tPutSlots.length; i++) tPutSlots[i] = i;
}
for (int tGrabSlot : tGrabSlots) {
byte tMovedItemCount = 0;
ItemStack tGrabStack = aTileEntity1.getStackInSlot(tGrabSlot);
if (listContainsItem(aFilter, tGrabStack, true, aInvertFilter)) {
if (tGrabStack.stackSize >= aMinMoveAtOnce
&& isAllowedToTakeFromSlot(aTileEntity1, tGrabSlot, aGrabFrom, tGrabStack)) {
for (int tPutSlot : tPutSlots) {
if (isAllowedToPutIntoSlot(
(IInventory) aTileEntity2, tPutSlot, aPutTo, tGrabStack, aMaxTargetStackSize)) {
tMovedItemCount += moveStackFromSlotAToSlotB(
aTileEntity1,
(IInventory) aTileEntity2,
tGrabSlot,
tPutSlot,
aMaxTargetStackSize,
aMinTargetStackSize,
(byte) (aMaxMoveAtOnce - tMovedItemCount),
aMinMoveAtOnce);
if (tMovedItemCount >= aMaxMoveAtOnce
|| (tMovedItemCount > 0 && aMaxTargetStackSize < 64)) return tMovedItemCount;
}
}
}
}
if (tMovedItemCount > 0) return tMovedItemCount;
}
if (aDoCheckChests && aTileEntity1 instanceof TileEntityChest) {
TileEntityChest tTileEntity1 = (TileEntityChest) aTileEntity1;
if (tTileEntity1.adjacentChestChecked) {
byte tAmount = 0;
if (tTileEntity1.adjacentChestXNeg != null) {
tAmount = moveOneItemStack(
tTileEntity1.adjacentChestXNeg,
aTileEntity2,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
false);
} else if (tTileEntity1.adjacentChestZNeg != null) {
tAmount = moveOneItemStack(
tTileEntity1.adjacentChestZNeg,
aTileEntity2,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
false);
} else if (tTileEntity1.adjacentChestXPos != null) {
tAmount = moveOneItemStack(
tTileEntity1.adjacentChestXPos,
aTileEntity2,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
false);
} else if (tTileEntity1.adjacentChestZPos != null) {
tAmount = moveOneItemStack(
tTileEntity1.adjacentChestZPos,
aTileEntity2,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
false);
}
if (tAmount != 0) return tAmount;
}
}
if (aDoCheckChests && aTileEntity2 instanceof TileEntityChest) {
TileEntityChest tTileEntity2 = (TileEntityChest) aTileEntity2;
if (tTileEntity2.adjacentChestChecked) {
byte tAmount = 0;
if (tTileEntity2.adjacentChestXNeg != null) {
tAmount = moveOneItemStack(
aTileEntity1,
tTileEntity2.adjacentChestXNeg,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
false);
} else if (tTileEntity2.adjacentChestZNeg != null) {
tAmount = moveOneItemStack(
aTileEntity1,
tTileEntity2.adjacentChestZNeg,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
false);
} else if (tTileEntity2.adjacentChestXPos != null) {
tAmount = moveOneItemStack(
aTileEntity1,
tTileEntity2.adjacentChestXPos,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
false);
} else if (tTileEntity2.adjacentChestZPos != null) {
tAmount = moveOneItemStack(
aTileEntity1,
tTileEntity2.adjacentChestZPos,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
false);
}
if (tAmount != 0) return tAmount;
}
}
}
return moveStackIntoPipe(
aTileEntity1,
aTileEntity2,
tGrabSlots,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
aDoCheckChests);
}
/**
* Moves Stack from Inv-Side to Inv-Slot.
*
* @return the Amount of moved Items
*/
public static byte moveOneItemStackIntoSlot(
Object aTileEntity1,
Object aTileEntity2,
byte aGrabFrom,
int aPutTo,
List aFilter,
boolean aInvertFilter,
byte aMaxTargetStackSize,
byte aMinTargetStackSize,
byte aMaxMoveAtOnce,
byte aMinMoveAtOnce) {
if (aTileEntity1 == null
|| !(aTileEntity1 instanceof IInventory)
|| aPutTo < 0
|| aMaxTargetStackSize <= 0
|| aMinTargetStackSize <= 0
|| aMaxMoveAtOnce <= 0
|| aMinTargetStackSize > aMaxTargetStackSize
|| aMinMoveAtOnce > aMaxMoveAtOnce) return 0;
int[] tGrabSlots = null;
if (aTileEntity1 instanceof ISidedInventory)
tGrabSlots = ((ISidedInventory) aTileEntity1).getAccessibleSlotsFromSide(aGrabFrom);
if (tGrabSlots == null) {
tGrabSlots = new int[((IInventory) aTileEntity1).getSizeInventory()];
for (int i = 0; i < tGrabSlots.length; i++) tGrabSlots[i] = i;
}
if (aTileEntity2 instanceof IInventory) {
for (int tGrabSlot : tGrabSlots) {
if (listContainsItem(
aFilter, ((IInventory) aTileEntity1).getStackInSlot(tGrabSlot), true, aInvertFilter)) {
if (isAllowedToTakeFromSlot(
(IInventory) aTileEntity1,
tGrabSlot,
aGrabFrom,
((IInventory) aTileEntity1).getStackInSlot(tGrabSlot))) {
if (isAllowedToPutIntoSlot(
(IInventory) aTileEntity2,
aPutTo,
(byte) 6,
((IInventory) aTileEntity1).getStackInSlot(tGrabSlot),
aMaxTargetStackSize)) {
byte tMovedItemCount = moveStackFromSlotAToSlotB(
(IInventory) aTileEntity1,
(IInventory) aTileEntity2,
tGrabSlot,
aPutTo,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce);
if (tMovedItemCount > 0) return tMovedItemCount;
}
}
}
}
}
moveStackIntoPipe(
((IInventory) aTileEntity1),
aTileEntity2,
tGrabSlots,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce);
return 0;
}
/**
* Moves Stack from Inv-Slot to Inv-Slot.
*
* @return the Amount of moved Items
*/
public static byte moveFromSlotToSlot(
IInventory aTileEntity1,
IInventory aTileEntity2,
int aGrabFrom,
int aPutTo,
List aFilter,
boolean aInvertFilter,
byte aMaxTargetStackSize,
byte aMinTargetStackSize,
byte aMaxMoveAtOnce,
byte aMinMoveAtOnce) {
if (aTileEntity1 == null
|| aTileEntity2 == null
|| aGrabFrom < 0
|| aPutTo < 0
|| aMaxTargetStackSize <= 0
|| aMinTargetStackSize <= 0
|| aMaxMoveAtOnce <= 0
|| aMinTargetStackSize > aMaxTargetStackSize
|| aMinMoveAtOnce > aMaxMoveAtOnce) return 0;
if (listContainsItem(aFilter, aTileEntity1.getStackInSlot(aGrabFrom), true, aInvertFilter)) {
if (isAllowedToTakeFromSlot(aTileEntity1, aGrabFrom, (byte) 6, aTileEntity1.getStackInSlot(aGrabFrom))) {
if (isAllowedToPutIntoSlot(
aTileEntity2, aPutTo, (byte) 6, aTileEntity1.getStackInSlot(aGrabFrom), aMaxTargetStackSize)) {
byte tMovedItemCount = moveStackFromSlotAToSlotB(
aTileEntity1,
aTileEntity2,
aGrabFrom,
aPutTo,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce);
if (tMovedItemCount > 0) return tMovedItemCount;
}
}
}
return 0;
}
/**
* Moves Stack from Inv-Side to Inv-Slot.
*
* @return the Amount of moved Items
*/
public static byte moveFromSlotToSide(
IInventory fromTile,
Object toTile,
int aGrabFrom,
byte aPutTo,
List aFilter,
boolean aInvertFilter,
byte aMaxTargetStackSize,
byte aMinTargetStackSize,
byte aMaxMoveAtOnce,
byte aMinMoveAtOnce,
boolean aDoCheckChests) {
if (fromTile == null
|| aGrabFrom < 0
|| aMinTargetStackSize <= 0
|| aMaxMoveAtOnce <= 0
|| aMinTargetStackSize > aMaxTargetStackSize
|| aMinMoveAtOnce > aMaxMoveAtOnce) return 0;
if (!listContainsItem(aFilter, fromTile.getStackInSlot(aGrabFrom), true, aInvertFilter)
|| !isAllowedToTakeFromSlot(fromTile, aGrabFrom, (byte) 6, fromTile.getStackInSlot(aGrabFrom)))
return 0;
if (toTile instanceof IInventory) {
int[] tPutSlots = null;
if (toTile instanceof ISidedInventory)
tPutSlots = ((ISidedInventory) toTile).getAccessibleSlotsFromSide(aPutTo);
if (tPutSlots == null) {
tPutSlots = new int[((IInventory) toTile).getSizeInventory()];
for (int i = 0; i < tPutSlots.length; i++) tPutSlots[i] = i;
}
byte tMovedItemCount = 0;
for (int tPutSlot : tPutSlots) {
if (isAllowedToPutIntoSlot(
(IInventory) toTile,
tPutSlot,
aPutTo,
fromTile.getStackInSlot(aGrabFrom),
aMaxTargetStackSize)) {
tMovedItemCount += moveStackFromSlotAToSlotB(
fromTile,
(IInventory) toTile,
aGrabFrom,
tPutSlot,
aMaxTargetStackSize,
aMinTargetStackSize,
(byte) (aMaxMoveAtOnce - tMovedItemCount),
aMinMoveAtOnce);
if (tMovedItemCount >= aMaxMoveAtOnce) {
return tMovedItemCount;
}
}
}
if (tMovedItemCount > 0) return tMovedItemCount;
if (aDoCheckChests && toTile instanceof TileEntityChest) {
TileEntityChest tTileEntity2 = (TileEntityChest) toTile;
if (tTileEntity2.adjacentChestChecked) {
if (tTileEntity2.adjacentChestXNeg != null) {
tMovedItemCount = moveFromSlotToSide(
fromTile,
tTileEntity2.adjacentChestXNeg,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
false);
} else if (tTileEntity2.adjacentChestZNeg != null) {
tMovedItemCount = moveFromSlotToSide(
fromTile,
tTileEntity2.adjacentChestZNeg,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
false);
} else if (tTileEntity2.adjacentChestXPos != null) {
tMovedItemCount = moveFromSlotToSide(
fromTile,
tTileEntity2.adjacentChestXPos,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
false);
} else if (tTileEntity2.adjacentChestZPos != null) {
tMovedItemCount = moveFromSlotToSide(
fromTile,
tTileEntity2.adjacentChestZPos,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
false);
}
if (tMovedItemCount > 0) return tMovedItemCount;
}
}
}
return moveStackIntoPipe(
fromTile,
toTile,
new int[] {aGrabFrom},
(byte) 6,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
aDoCheckChests);
}
public static byte moveFromSlotToSide(
IInventory fromTile,
Object toTile,
int aGrabFrom,
byte aPutTo,
List aFilter,
boolean aInvertFilter,
byte aMaxTargetStackSize,
byte aMinTargetStackSize,
byte aMaxMoveAtOnce,
byte aMinMoveAtOnce) {
return moveFromSlotToSide(
fromTile,
toTile,
aGrabFrom,
aPutTo,
aFilter,
aInvertFilter,
aMaxTargetStackSize,
aMinTargetStackSize,
aMaxMoveAtOnce,
aMinMoveAtOnce,
true);
}
public static boolean listContainsItem(
Collection aList, ItemStack aStack, boolean aTIfListEmpty, boolean aInvertFilter) {
if (aStack == null || aStack.stackSize < 1) return false;
if (aList == null) return aTIfListEmpty;
boolean tEmpty = true;
for (ItemStack tStack : aList) {
if (tStack != null) {
tEmpty = false;
if (areStacksEqual(aStack, tStack)) {
return !aInvertFilter;
}
}
}
return tEmpty ? aTIfListEmpty : aInvertFilter;
}
public static boolean areStacksOrToolsEqual(ItemStack aStack1, ItemStack aStack2) {
if (aStack1 != null && aStack2 != null && aStack1.getItem() == aStack2.getItem()) {
if (aStack1.getItem().isDamageable()) return true;
return ((aStack1.getTagCompound() == null) == (aStack2.getTagCompound() == null))
&& (aStack1.getTagCompound() == null
|| aStack1.getTagCompound().equals(aStack2.getTagCompound()))
&& (Items.feather.getDamage(aStack1) == Items.feather.getDamage(aStack2)
|| Items.feather.getDamage(aStack1) == W
|| Items.feather.getDamage(aStack2) == W);
}
return false;
}
public static boolean areFluidsEqual(FluidStack aFluid1, FluidStack aFluid2) {
return areFluidsEqual(aFluid1, aFluid2, false);
}
public static boolean areFluidsEqual(FluidStack aFluid1, FluidStack aFluid2, boolean aIgnoreNBT) {
return aFluid1 != null
&& aFluid2 != null
&& aFluid1.getFluid() == aFluid2.getFluid()
&& (aIgnoreNBT
|| ((aFluid1.tag == null) == (aFluid2.tag == null))
&& (aFluid1.tag == null || aFluid1.tag.equals(aFluid2.tag)));
}
public static boolean areStacksEqual(ItemStack aStack1, ItemStack aStack2) {
return areStacksEqual(aStack1, aStack2, false);
}
public static boolean areStacksEqual(ItemStack aStack1, ItemStack aStack2, boolean aIgnoreNBT) {
return aStack1 != null
&& aStack2 != null
&& aStack1.getItem() == aStack2.getItem()
&& (aIgnoreNBT
|| (((aStack1.getTagCompound() == null) == (aStack2.getTagCompound() == null))
&& (aStack1.getTagCompound() == null
|| aStack1.getTagCompound().equals(aStack2.getTagCompound()))))
&& (Items.feather.getDamage(aStack1) == Items.feather.getDamage(aStack2)
|| Items.feather.getDamage(aStack1) == W
|| Items.feather.getDamage(aStack2) == W);
}
/**
* Treat both null list, or both null item stack at same list position as equal.
*
* Since ItemStack doesn't override equals and hashCode, you cannot just use Objects.equals
*/
public static boolean areStackListsEqual(
List lhs, List rhs, boolean ignoreStackSize, boolean ignoreNBT) {
if (lhs == null) return rhs == null;
if (rhs == null) return false;
if (lhs.size() != rhs.size()) return false;
for (Iterator it1 = lhs.iterator(), it2 = rhs.iterator(); it1.hasNext() && it2.hasNext(); ) {
if (!areStacksEqualExtended(it1.next(), it2.next(), ignoreStackSize, ignoreNBT)) return false;
}
return true;
}
private static boolean areStacksEqualExtended(
ItemStack lhs, ItemStack rhs, boolean ignoreStackSize, boolean ignoreNBT) {
if (lhs == null) return rhs == null;
if (rhs == null) return false;
return lhs.getItem() == rhs.getItem()
&& (ignoreNBT || Objects.equals(lhs.stackTagCompound, rhs.stackTagCompound))
&& (ignoreStackSize || lhs.stackSize == rhs.stackSize);
}
public static boolean areUnificationsEqual(ItemStack aStack1, ItemStack aStack2) {
return areUnificationsEqual(aStack1, aStack2, false);
}
public static boolean areUnificationsEqual(ItemStack aStack1, ItemStack aStack2, boolean aIgnoreNBT) {
return areStacksEqual(
GT_OreDictUnificator.get_nocopy(aStack1), GT_OreDictUnificator.get_nocopy(aStack2), aIgnoreNBT);
}
public static String getFluidName(Fluid aFluid, boolean aLocalized) {
if (aFluid == null) return E;
String rName = aLocalized ? aFluid.getLocalizedName(new FluidStack(aFluid, 0)) : aFluid.getUnlocalizedName();
if (rName.contains("fluid.") || rName.contains("tile."))
return capitalizeString(rName.replaceAll("fluid.", E).replaceAll("tile.", E));
return rName;
}
public static String getFluidName(FluidStack aFluid, boolean aLocalized) {
if (aFluid == null) return E;
return getFluidName(aFluid.getFluid(), aLocalized);
}
public static void reInit() {
sFilledContainerToData.clear();
sEmptyContainerToFluidToData.clear();
sFluidToContainers.clear();
sFluidUnlocalizedNameToFluid.clear();
for (FluidContainerData tData : sFluidContainerList) {
sFilledContainerToData.put(new GT_ItemStack(tData.filledContainer), tData);
Map tFluidToContainer =
sEmptyContainerToFluidToData.get(new GT_ItemStack(tData.emptyContainer));
List tContainers = sFluidToContainers.get(tData.fluid.getFluid());
if (tFluidToContainer == null) {
sEmptyContainerToFluidToData.put(
new GT_ItemStack(tData.emptyContainer), tFluidToContainer = new /*Concurrent*/ HashMap<>());
GregTech_API.sFluidMappings.add(tFluidToContainer);
}
tFluidToContainer.put(tData.fluid.getFluid(), tData);
if (tContainers == null) {
tContainers = new ArrayList<>();
tContainers.add(tData.filledContainer);
sFluidToContainers.put(tData.fluid.getFluid(), tContainers);
} else tContainers.add(tData.filledContainer);
}
for (Fluid tFluid : FluidRegistry.getRegisteredFluids().values()) {
sFluidUnlocalizedNameToFluid.put(tFluid.getUnlocalizedName(), tFluid);
}
}
public static Fluid getFluidFromUnlocalizedName(String aUnlocalizedName) {
return sFluidUnlocalizedNameToFluid.get(aUnlocalizedName);
}
public static void addFluidContainerData(FluidContainerData aData) {
sFluidContainerList.add(aData);
sFilledContainerToData.put(new GT_ItemStack(aData.filledContainer), aData);
Map tFluidToContainer =
sEmptyContainerToFluidToData.get(new GT_ItemStack(aData.emptyContainer));
List tContainers = sFluidToContainers.get(aData.fluid.getFluid());
if (tFluidToContainer == null) {
sEmptyContainerToFluidToData.put(
new GT_ItemStack(aData.emptyContainer), tFluidToContainer = new /*Concurrent*/ HashMap<>());
GregTech_API.sFluidMappings.add(tFluidToContainer);
}
tFluidToContainer.put(aData.fluid.getFluid(), aData);
if (tContainers == null) {
tContainers = new ArrayList<>();
tContainers.add(aData.filledContainer);
sFluidToContainers.put(aData.fluid.getFluid(), tContainers);
} else tContainers.add(aData.filledContainer);
}
public static List getContainersFromFluid(FluidStack tFluidStack) {
if (tFluidStack != null) {
List tContainers = sFluidToContainers.get(tFluidStack.getFluid());
if (tContainers == null) return new ArrayList<>();
return tContainers;
}
return new ArrayList<>();
}
public static ItemStack fillFluidContainer(
FluidStack aFluid, ItemStack aStack, boolean aRemoveFluidDirectly, boolean aCheckIFluidContainerItems) {
if (isStackInvalid(aStack) || aFluid == null) return null;
if (GT_ModHandler.isWater(aFluid) && ItemList.Bottle_Empty.isStackEqual(aStack)) {
if (aFluid.amount >= 250) {
if (aRemoveFluidDirectly) aFluid.amount -= 250;
return new ItemStack(Items.potionitem, 1, 0);
}
return null;
}
if (aCheckIFluidContainerItems
&& aStack.getItem() instanceof IFluidContainerItem
&& ((IFluidContainerItem) aStack.getItem()).getFluid(aStack) == null
&& ((IFluidContainerItem) aStack.getItem()).getCapacity(aStack) <= aFluid.amount) {
if (aRemoveFluidDirectly)
aFluid.amount -=
((IFluidContainerItem) aStack.getItem()).fill(aStack = copyAmount(1, aStack), aFluid, true);
else ((IFluidContainerItem) aStack.getItem()).fill(aStack = copyAmount(1, aStack), aFluid, true);
return aStack;
}
Map tFluidToContainer = sEmptyContainerToFluidToData.get(new GT_ItemStack(aStack));
if (tFluidToContainer == null) return null;
FluidContainerData tData = tFluidToContainer.get(aFluid.getFluid());
if (tData == null || tData.fluid.amount > aFluid.amount) return null;
if (aRemoveFluidDirectly) aFluid.amount -= tData.fluid.amount;
return copyAmount(1, tData.filledContainer);
}
public static ItemStack getFluidDisplayStack(Fluid aFluid) {
return aFluid == null ? null : getFluidDisplayStack(new FluidStack(aFluid, 0), false);
}
public static ItemStack getFluidDisplayStack(FluidStack aFluid, boolean aUseStackSize) {
return getFluidDisplayStack(aFluid, aUseStackSize, false);
}
public static ItemStack getFluidDisplayStack(FluidStack aFluid, boolean aUseStackSize, boolean aHideStackSize) {
if (aFluid == null || aFluid.getFluid() == null) return null;
int tmp = 0;
try {
tmp = aFluid.getFluid().getID();
} catch (Exception e) {
System.err.println(e);
}
ItemStack rStack = ItemList.Display_Fluid.getWithDamage(1, tmp);
NBTTagCompound tNBT = new NBTTagCompound();
tNBT.setLong("mFluidDisplayAmount", aUseStackSize ? aFluid.amount : 0);
tNBT.setLong("mFluidDisplayHeat", aFluid.getFluid().getTemperature(aFluid));
tNBT.setBoolean("mFluidState", aFluid.getFluid().isGaseous(aFluid));
tNBT.setBoolean("mHideStackSize", aHideStackSize);
rStack.setTagCompound(tNBT);
return rStack;
}
public static FluidStack getFluidFromDisplayStack(ItemStack aDisplayStack) {
if (!isStackValid(aDisplayStack)
|| aDisplayStack.getItem() != ItemList.Display_Fluid.getItem()
|| !aDisplayStack.hasTagCompound()) return null;
Fluid tFluid = FluidRegistry.getFluid(ItemList.Display_Fluid.getItem().getDamage(aDisplayStack));
return new FluidStack(tFluid, (int) aDisplayStack.getTagCompound().getLong("mFluidDisplayAmount"));
}
public static boolean containsFluid(ItemStack aStack, FluidStack aFluid, boolean aCheckIFluidContainerItems) {
if (isStackInvalid(aStack) || aFluid == null) return false;
if (aCheckIFluidContainerItems
&& aStack.getItem() instanceof IFluidContainerItem
&& ((IFluidContainerItem) aStack.getItem()).getCapacity(aStack) > 0)
return aFluid.isFluidEqual(
((IFluidContainerItem) aStack.getItem()).getFluid(aStack = copyAmount(1, aStack)));
FluidContainerData tData = sFilledContainerToData.get(new GT_ItemStack(aStack));
return tData != null && tData.fluid.isFluidEqual(aFluid);
}
public static FluidStack getFluidForFilledItem(ItemStack aStack, boolean aCheckIFluidContainerItems) {
if (isStackInvalid(aStack)) return null;
if (aCheckIFluidContainerItems
&& aStack.getItem() instanceof IFluidContainerItem
&& ((IFluidContainerItem) aStack.getItem()).getCapacity(aStack) > 0)
return ((IFluidContainerItem) aStack.getItem()).drain(copyAmount(1, aStack), Integer.MAX_VALUE, true);
FluidContainerData tData = sFilledContainerToData.get(new GT_ItemStack(aStack));
return tData == null ? null : tData.fluid.copy();
}
/**
* Get empty fluid container from filled one.
*/
public static ItemStack getContainerForFilledItem(ItemStack aStack, boolean aCheckIFluidContainerItems) {
if (isStackInvalid(aStack)) return null;
FluidContainerData tData = sFilledContainerToData.get(new GT_ItemStack(aStack));
if (tData != null) return copyAmount(1, tData.emptyContainer);
if (aCheckIFluidContainerItems
&& aStack.getItem() instanceof IFluidContainerItem
&& ((IFluidContainerItem) aStack.getItem()).getCapacity(aStack) > 0) {
((IFluidContainerItem) aStack.getItem()).drain(aStack = copyAmount(1, aStack), Integer.MAX_VALUE, true);
return aStack;
}
return null;
}
/**
* Get general container item, not only fluid container but also non-consumable item.
* {@link #getContainerForFilledItem} works better for fluid container.
*/
public static ItemStack getContainerItem(ItemStack aStack, boolean aCheckIFluidContainerItems) {
if (isStackInvalid(aStack)) return null;
if (aStack.getItem().hasContainerItem(aStack)) return aStack.getItem().getContainerItem(aStack);
/** These are all special Cases, in which it is intended to have only GT Blocks outputting those Container Items */
if (ItemList.Cell_Empty.isStackEqual(aStack, false, true)) return null;
if (aStack.getItem() == Items.potionitem
|| aStack.getItem() == Items.experience_bottle
|| ItemList.TF_Vial_FieryBlood.isStackEqual(aStack)
|| ItemList.TF_Vial_FieryTears.isStackEqual(aStack)) return ItemList.Bottle_Empty.get(1);
if (aCheckIFluidContainerItems
&& aStack.getItem() instanceof IFluidContainerItem
&& ((IFluidContainerItem) aStack.getItem()).getCapacity(aStack) > 0) {
ItemStack tStack = copyAmount(1, aStack);
((IFluidContainerItem) aStack.getItem()).drain(tStack, Integer.MAX_VALUE, true);
if (!areStacksEqual(aStack, tStack)) return tStack;
return null;
}
int tCapsuleCount = GT_ModHandler.getCapsuleCellContainerCount(aStack);
if (tCapsuleCount > 0) return ItemList.Cell_Empty.get(tCapsuleCount);
if (ItemList.IC2_ForgeHammer.isStackEqual(aStack) || ItemList.IC2_WireCutter.isStackEqual(aStack))
return copyMetaData(Items.feather.getDamage(aStack) + 1, aStack);
return null;
}
public static FluidStack getFluidFromContainerOrFluidDisplay(ItemStack stack) {
FluidStack fluidStack = GT_Utility.getFluidForFilledItem(stack, true);
if (fluidStack == null) {
fluidStack = GT_Utility.getFluidFromDisplayStack(stack);
}
return fluidStack;
}
public static synchronized boolean removeIC2BottleRecipe(
ItemStack aContainer,
ItemStack aInput,
Map aRecipeList,
ItemStack aOutput) {
if ((isStackInvalid(aInput) && isStackInvalid(aOutput) && isStackInvalid(aContainer)) || aRecipeList == null)
return false;
boolean rReturn = false;
Iterator> tIterator =
aRecipeList.entrySet().iterator();
aOutput = GT_OreDictUnificator.get(aOutput);
while (tIterator.hasNext()) {
Map.Entry tEntry = tIterator.next();
if (aInput == null || tEntry.getKey().matches(aContainer, aInput)) {
List tList = tEntry.getValue().items;
if (tList != null)
for (ItemStack tOutput : tList)
if (aOutput == null || areStacksEqual(GT_OreDictUnificator.get(tOutput), aOutput)) {
tIterator.remove();
rReturn = true;
break;
}
}
}
return rReturn;
}
public static synchronized boolean removeSimpleIC2MachineRecipe(
ItemStack aInput, Map aRecipeList, ItemStack aOutput) {
if ((isStackInvalid(aInput) && isStackInvalid(aOutput)) || aRecipeList == null) return false;
boolean rReturn = false;
Iterator> tIterator =
aRecipeList.entrySet().iterator();
aOutput = GT_OreDictUnificator.get(aOutput);
while (tIterator.hasNext()) {
Map.Entry tEntry = tIterator.next();
if (aInput == null || tEntry.getKey().matches(aInput)) {
List tList = tEntry.getValue().items;
if (tList != null)
for (ItemStack tOutput : tList)
if (aOutput == null || areStacksEqual(GT_OreDictUnificator.get(tOutput), aOutput)) {
tIterator.remove();
rReturn = true;
break;
}
}
}
return rReturn;
}
public static synchronized void bulkRemoveSimpleIC2MachineRecipe(
Map toRemove, Map aRecipeList) {
if (aRecipeList == null || aRecipeList.isEmpty()) return;
toRemove.entrySet().removeIf(aEntry -> (isStackInvalid(aEntry.getKey()) && isStackInvalid(aEntry.getValue())));
final Map finalToRemove =
Maps.transformValues(toRemove, GT_OreDictUnificator::get_nocopy);
aRecipeList.entrySet().removeIf(tEntry -> finalToRemove.entrySet().stream()
.anyMatch(aEntry -> {
final ItemStack aInput = aEntry.getKey(), aOutput = aEntry.getValue();
final List tList = tEntry.getValue().items;
if (tList == null) return false;
if (aInput != null && !tEntry.getKey().matches(aInput)) return false;
return tList.stream()
.anyMatch(tOutput ->
(aOutput == null || areStacksEqual(GT_OreDictUnificator.get(tOutput), aOutput)));
}));
}
public static boolean addSimpleIC2MachineRecipe(
ItemStack aInput, Map aRecipeList, NBTTagCompound aNBT, Object... aOutput) {
if (isStackInvalid(aInput) || aOutput.length == 0 || aRecipeList == null) return false;
ItemData tOreName = GT_OreDictUnificator.getAssociation(aInput);
for (Object o : aOutput) {
if (o == null) {
GT_FML_LOGGER.info("EmptyIC2Output!" + aInput.getUnlocalizedName());
return false;
}
}
ItemStack[] tStack = GT_OreDictUnificator.getStackArray(true, aOutput);
if (tStack.length > 0 && areStacksEqual(aInput, tStack[0])) return false;
if (tOreName != null) {
if (tOreName.toString().equals("dustAsh")
&& tStack[0].getUnlocalizedName().equals("tile.volcanicAsh")) return false;
aRecipeList.put(
new RecipeInputOreDict(tOreName.toString(), aInput.stackSize), new RecipeOutput(aNBT, tStack));
} else {
aRecipeList.put(
new RecipeInputItemStack(copyOrNull(aInput), aInput.stackSize), new RecipeOutput(aNBT, tStack));
}
return true;
}
public static ItemStack getWrittenBook(String aMapping, ItemStack aStackToPutNBT) {
if (isStringInvalid(aMapping)) return null;
ItemStack rStack = GregTech_API.sBookList.get(aMapping);
if (rStack == null) return aStackToPutNBT;
if (aStackToPutNBT != null) {
aStackToPutNBT.setTagCompound(rStack.getTagCompound());
return aStackToPutNBT;
}
return copyAmount(1, rStack);
}
public static ItemStack getWrittenBook(String aMapping, String aTitle, String aAuthor, String... aPages) {
if (isStringInvalid(aMapping)) return null;
ItemStack rStack = GregTech_API.sBookList.get(aMapping);
if (rStack != null) return copyAmount(1, rStack);
if (isStringInvalid(aTitle) || isStringInvalid(aAuthor) || aPages.length <= 0) return null;
sBookCount++;
rStack = new ItemStack(Items.written_book, 1);
NBTTagCompound tNBT = new NBTTagCompound();
tNBT.setString("title", GT_LanguageManager.addStringLocalization("Book." + aTitle + ".Name", aTitle));
tNBT.setString("author", aAuthor);
NBTTagList tNBTList = new NBTTagList();
for (byte i = 0; i < aPages.length; i++) {
aPages[i] = GT_LanguageManager.addStringLocalization(
"Book." + aTitle + ".Page" + ((i < 10) ? "0" + i : i), aPages[i]);
if (i < 48) {
if (aPages[i].length() < 256) tNBTList.appendTag(new NBTTagString(aPages[i]));
else GT_Log.err.println("WARNING: String for written Book too long! -> " + aPages[i]);
} else {
GT_Log.err.println("WARNING: Too much Pages for written Book! -> " + aTitle);
break;
}
}
tNBTList.appendTag(new NBTTagString("Credits to " + aAuthor + " for writing this Book. This was Book Nr. "
+ sBookCount + " at its creation. Gotta get 'em all!"));
tNBT.setTag("pages", tNBTList);
rStack.setTagCompound(tNBT);
GT_Log.out.println("GT_Mod: Added Book to Book List - Mapping: '" + aMapping + "' - Name: '" + aTitle
+ "' - Author: '" + aAuthor + "'");
GregTech_API.sBookList.put(aMapping, rStack);
return copyOrNull(rStack);
}
public static boolean doSoundAtClient(String aSoundName, int aTimeUntilNextSound, float aSoundStrength) {
if (aSoundName == null) return false;
return doSoundAtClient(aSoundName, aTimeUntilNextSound, aSoundStrength, GT.getThePlayer());
}
public static boolean doSoundAtClient(SoundResource sound, int aTimeUntilNextSound, float aSoundStrength) {
return doSoundAtClient(sound.resourceLocation, aTimeUntilNextSound, aSoundStrength, GT.getThePlayer());
}
public static boolean doSoundAtClient(
ResourceLocation aSoundResourceLocation, int aTimeUntilNextSound, float aSoundStrength) {
return doSoundAtClient(aSoundResourceLocation, aTimeUntilNextSound, aSoundStrength, GT.getThePlayer());
}
public static boolean doSoundAtClient(
String aSoundName, int aTimeUntilNextSound, float aSoundStrength, Entity aEntity) {
if (aEntity == null || aSoundName == null) return false;
return doSoundAtClient(
aSoundName, aTimeUntilNextSound, aSoundStrength, aEntity.posX, aEntity.posY, aEntity.posZ);
}
public static boolean doSoundAtClient(
ResourceLocation aSoundResourceLocation, int aTimeUntilNextSound, float aSoundStrength, Entity aEntity) {
if (aEntity == null) return false;
return doSoundAtClient(
aSoundResourceLocation.toString(),
aTimeUntilNextSound,
aSoundStrength,
aEntity.posX,
aEntity.posY,
aEntity.posZ);
}
public static boolean doSoundAtClient(
ResourceLocation aSoundResourceLocation,
int aTimeUntilNextSound,
float aSoundStrength,
double aX,
double aY,
double aZ) {
return doSoundAtClient(aSoundResourceLocation, aTimeUntilNextSound, aSoundStrength, 1.01818028F, aX, aY, aZ);
}
/**
* @inheritDoc
* @deprecated Use {@link #doSoundAtClient(ResourceLocation, int, float, double, double, double)}
*/
@Deprecated
public static boolean doSoundAtClient(
String aSoundName, int aTimeUntilNextSound, float aSoundStrength, double aX, double aY, double aZ) {
if (aSoundName == null) return false;
return doSoundAtClient(
new ResourceLocation(aSoundName), aTimeUntilNextSound, aSoundStrength, 1.01818028F, aX, aY, aZ);
}
public static boolean doSoundAtClient(
SoundResource aSound, int aTimeUntilNextSound, float aSoundStrength, double aX, double aY, double aZ) {
return doSoundAtClient(aSound.resourceLocation, aTimeUntilNextSound, aSoundStrength, aX, aY, aZ);
}
public static boolean doSoundAtClient(
SoundResource aSound,
int aTimeUntilNextSound,
float aSoundStrength,
float aSoundModulation,
double aX,
double aY,
double aZ) {
return doSoundAtClient(
aSound.resourceLocation, aTimeUntilNextSound, aSoundStrength, aSoundModulation, aX, aY, aZ);
}
public static boolean doSoundAtClient(
ResourceLocation aSoundResourceLocation,
int aTimeUntilNextSound,
float aSoundStrength,
float aSoundModulation,
double aX,
double aY,
double aZ) {
if (!FMLCommonHandler.instance().getEffectiveSide().isClient()
|| GT.getThePlayer() == null
|| !GT.getThePlayer().worldObj.isRemote) return false;
if (GregTech_API.sMultiThreadedSounds)
new Thread(
new GT_Runnable_Sound(
GT.getThePlayer().worldObj,
MathHelper.floor_double(aX),
MathHelper.floor_double(aY),
MathHelper.floor_double(aZ),
aTimeUntilNextSound,
aSoundResourceLocation,
aSoundStrength,
aSoundModulation),
"Sound Effect")
.start();
else
new GT_Runnable_Sound(
GT.getThePlayer().worldObj,
MathHelper.floor_double(aX),
MathHelper.floor_double(aY),
MathHelper.floor_double(aZ),
aTimeUntilNextSound,
aSoundResourceLocation,
aSoundStrength,
aSoundModulation)
.run();
return true;
}
/**
* @inheritDoc
* @Deprecated Use {@link #doSoundAtClient(ResourceLocation, int, float, float, double, double, double)}
*/
@Deprecated
public static boolean doSoundAtClient(
String aSoundName,
int aTimeUntilNextSound,
float aSoundStrength,
float aSoundModulation,
double aX,
double aY,
double aZ) {
if (isStringInvalid(aSoundName)) return false;
return doSoundAtClient(
new ResourceLocation(aSoundName), aTimeUntilNextSound, aSoundStrength, aSoundModulation, aX, aY, aZ);
}
public static boolean sendSoundToPlayers(
World aWorld, String aSoundName, float aSoundStrength, float aSoundModulation, int aX, int aY, int aZ) {
if (isStringInvalid(aSoundName) || aWorld == null || aWorld.isRemote) return false;
NW.sendPacketToAllPlayersInRange(
aWorld, new GT_Packet_Sound(aSoundName, aSoundStrength, aSoundModulation, aX, (short) aY, aZ), aX, aZ);
return true;
}
public static boolean sendSoundToPlayers(
World aWorld, SoundResource sound, float aSoundStrength, float aSoundModulation, int aX, int aY, int aZ) {
if (aWorld == null || aWorld.isRemote) return false;
NW.sendPacketToAllPlayersInRange(
aWorld,
new GT_Packet_Sound(
sound.resourceLocation.toString(), aSoundStrength, aSoundModulation, aX, (short) aY, aZ),
aX,
aZ);
return true;
}
public static int stackToInt(ItemStack aStack) {
if (isStackInvalid(aStack)) return 0;
return itemToInt(aStack.getItem(), Items.feather.getDamage(aStack));
}
public static int itemToInt(Item aItem, int aMeta) {
return Item.getIdFromItem(aItem) | (aMeta << 16);
}
public static int stackToWildcard(ItemStack aStack) {
if (isStackInvalid(aStack)) return 0;
return Item.getIdFromItem(aStack.getItem()) | (W << 16);
}
public static ItemStack intToStack(int aStack) {
int tID = aStack & (~0 >>> 16), tMeta = aStack >>> 16;
Item tItem = Item.getItemById(tID);
if (tItem != null) return new ItemStack(tItem, 1, tMeta);
return null;
}
public static Integer[] stacksToIntegerArray(ItemStack... aStacks) {
Integer[] rArray = new Integer[aStacks.length];
for (int i = 0; i < rArray.length; i++) {
rArray[i] = stackToInt(aStacks[i]);
}
return rArray;
}
public static int[] stacksToIntArray(ItemStack... aStacks) {
int[] rArray = new int[aStacks.length];
for (int i = 0; i < rArray.length; i++) {
rArray[i] = stackToInt(aStacks[i]);
}
return rArray;
}
public static boolean arrayContains(Object aObject, Object... aObjects) {
return listContains(aObject, Arrays.asList(aObjects));
}
public static boolean listContains(Object aObject, Collection aObjects) {
if (aObjects == null) return false;
return aObjects.contains(aObject);
}
public static boolean arrayContainsNonNull(T... aArray) {
if (aArray != null) for (Object tObject : aArray) if (tObject != null) return true;
return false;
}
/**
* Note: use {@link ArrayExt#withoutNulls(Object[], IntFunction)} if you want an array as a result.
*/
public static ArrayList getArrayListWithoutNulls(T... aArray) {
if (aArray == null) return new ArrayList<>();
ArrayList rList = new ArrayList<>(Arrays.asList(aArray));
for (int i = 0; i < rList.size(); i++) if (rList.get(i) == null) rList.remove(i--);
return rList;
}
/**
* Note: use {@link ArrayExt#withoutTrailingNulls(Object[], IntFunction)} if you want an array as a result.
*/
public static ArrayList getArrayListWithoutTrailingNulls(T... aArray) {
if (aArray == null) return new ArrayList<>();
ArrayList rList = new ArrayList<>(Arrays.asList(aArray));
for (int i = rList.size() - 1; i >= 0 && rList.get(i) == null; ) rList.remove(i--);
return rList;
}
@Deprecated // why do you use Objects?
public static Block getBlock(Object aBlock) {
return (Block) aBlock;
}
public static Block getBlockFromStack(ItemStack itemStack) {
if (isStackInvalid(itemStack)) return Blocks.air;
return getBlockFromItem(itemStack.getItem());
}
public static Block getBlockFromItem(Item item) {
return Block.getBlockFromItem(item);
}
@Deprecated // why do you use Objects? And if you want to check your block to be not null, check it directly!
public static boolean isBlockValid(Object aBlock) {
return (aBlock instanceof Block);
}
@Deprecated // why do you use Objects? And if you want to check your block to be null, check it directly!
public static boolean isBlockInvalid(Object aBlock) {
return !(aBlock instanceof Block);
}
public static boolean isStringValid(Object aString) {
return aString != null && !aString.toString().isEmpty();
}
public static boolean isStringInvalid(Object aString) {
return aString == null || aString.toString().isEmpty();
}
public static boolean isStackValid(Object aStack) {
return (aStack instanceof ItemStack)
&& ((ItemStack) aStack).getItem() != null
&& ((ItemStack) aStack).stackSize >= 0;
}
public static boolean isStackInvalid(Object aStack) {
return aStack == null
|| !(aStack instanceof ItemStack)
|| ((ItemStack) aStack).getItem() == null
|| ((ItemStack) aStack).stackSize < 0;
}
public static boolean isDebugItem(ItemStack aStack) {
return /*ItemList.Armor_Cheat.isStackEqual(aStack, T, T) || */ areStacksEqual(
GT_ModHandler.getIC2Item("debug", 1), aStack, true);
}
public static ItemStack updateItemStack(ItemStack aStack) {
if (isStackValid(aStack) && aStack.getItem() instanceof GT_Generic_Item)
((GT_Generic_Item) aStack.getItem()).isItemStackUsable(aStack);
return aStack;
}
public static boolean isOpaqueBlock(World aWorld, int aX, int aY, int aZ) {
return aWorld.getBlock(aX, aY, aZ).isOpaqueCube();
}
public static boolean isBlockAir(World aWorld, int aX, int aY, int aZ) {
return aWorld.getBlock(aX, aY, aZ).isAir(aWorld, aX, aY, aZ);
}
public static boolean hasBlockHitBox(World aWorld, int aX, int aY, int aZ) {
return aWorld.getBlock(aX, aY, aZ).getCollisionBoundingBoxFromPool(aWorld, aX, aY, aZ) != null;
}
public static void setCoordsOnFire(World aWorld, int aX, int aY, int aZ, boolean aReplaceCenter) {
if (aReplaceCenter)
if (aWorld.getBlock(aX, aY, aZ).getCollisionBoundingBoxFromPool(aWorld, aX, aY, aZ) == null)
aWorld.setBlock(aX, aY, aZ, Blocks.fire);
if (aWorld.getBlock(aX + 1, aY, aZ).getCollisionBoundingBoxFromPool(aWorld, aX + 1, aY, aZ) == null)
aWorld.setBlock(aX + 1, aY, aZ, Blocks.fire);
if (aWorld.getBlock(aX - 1, aY, aZ).getCollisionBoundingBoxFromPool(aWorld, aX - 1, aY, aZ) == null)
aWorld.setBlock(aX - 1, aY, aZ, Blocks.fire);
if (aWorld.getBlock(aX, aY + 1, aZ).getCollisionBoundingBoxFromPool(aWorld, aX, aY + 1, aZ) == null)
aWorld.setBlock(aX, aY + 1, aZ, Blocks.fire);
if (aWorld.getBlock(aX, aY - 1, aZ).getCollisionBoundingBoxFromPool(aWorld, aX, aY - 1, aZ) == null)
aWorld.setBlock(aX, aY - 1, aZ, Blocks.fire);
if (aWorld.getBlock(aX, aY, aZ + 1).getCollisionBoundingBoxFromPool(aWorld, aX, aY, aZ + 1) == null)
aWorld.setBlock(aX, aY, aZ + 1, Blocks.fire);
if (aWorld.getBlock(aX, aY, aZ - 1).getCollisionBoundingBoxFromPool(aWorld, aX, aY, aZ - 1) == null)
aWorld.setBlock(aX, aY, aZ - 1, Blocks.fire);
}
public static ItemStack getProjectile(SubTag aProjectileType, IInventory aInventory) {
if (aInventory != null)
for (int i = 0, j = aInventory.getSizeInventory(); i < j; i++) {
ItemStack rStack = aInventory.getStackInSlot(i);
if (isStackValid(rStack)
&& rStack.getItem() instanceof IProjectileItem
&& ((IProjectileItem) rStack.getItem()).hasProjectile(aProjectileType, rStack))
return updateItemStack(rStack);
}
return null;
}
public static void removeNullStacksFromInventory(IInventory aInventory) {
if (aInventory != null)
for (int i = 0, j = aInventory.getSizeInventory(); i < j; i++) {
ItemStack tStack = aInventory.getStackInSlot(i);
if (tStack != null && (tStack.stackSize == 0 || tStack.getItem() == null))
aInventory.setInventorySlotContents(i, null);
}
}
/**
* Initializes new empty texture page for casings
* page 0 is old CASING_BLOCKS
*
* Then casings should be registered like this:
* for (byte i = MIN_USED_META; i < MAX_USED_META; i = (byte) (i + 1)) {
* Textures.BlockIcons.casingTexturePages[PAGE][i+START_INDEX] = new GT_CopiedBlockTexture(this, 6, i);
* }
*
* @param page 0 to 127
* @return true if it made empty page, false if one already existed...
*/
public static boolean addTexturePage(byte page) {
if (Textures.BlockIcons.casingTexturePages[page] == null) {
Textures.BlockIcons.casingTexturePages[page] = new ITexture[128];
return true;
}
return false;
}
/**
* Return texture id from page and index, for use when determining hatches, but can also be precomputed from:
* (page<<7)+index
* @param page 0 to 127 page
* @param index 0 to 127 texture index
* @return casing texture 0 to 16383
*/
public static int getTextureId(byte page, byte index) {
if (page >= 0 && index >= 0) {
return (page << 7) + index;
}
throw new RuntimeException("Index out of range: [" + page + "][" + index + "]");
}
/**
* Return texture id from page and index, for use when determining hatches, but can also be precomputed from:
* (page<<7)+index
* @param page 0 to 127 page
* @param startIndex 0 to 127 start texture index
* @param blockMeta meta of the block
* @return casing texture 0 to 16383
*/
public static int getTextureId(byte page, byte startIndex, byte blockMeta) {
if (page >= 0 && startIndex >= 0 && blockMeta >= 0 && (startIndex + blockMeta) <= 127) {
return (page << 7) + (startIndex + blockMeta);
}
throw new RuntimeException("Index out of range: [" + page + "][" + startIndex + "+" + blockMeta + "="
+ (startIndex + blockMeta) + "]");
}
/**
* Return texture id from item stack, unoptimized but readable?
* @return casing texture 0 to 16383
*/
@Deprecated
public static int getTextureId(ItemStack stack) {
return getTextureId(Block.getBlockFromItem(stack.getItem()), (byte) stack.getItemDamage());
}
/**
* Return texture id from item stack, unoptimized but readable?
* @return casing texture 0 to 16383
*/
public static int getTextureId(Block blockFromBlock, byte metaFromBlock) {
for (int page = 0; page < Textures.BlockIcons.casingTexturePages.length; page++) {
ITexture[] casingTexturePage = Textures.BlockIcons.casingTexturePages[page];
if (casingTexturePage != null) {
for (int index = 0; index < casingTexturePage.length; index++) {
ITexture iTexture = casingTexturePage[index];
if (iTexture instanceof IBlockContainer) {
Block block = ((IBlockContainer) iTexture).getBlock();
byte meta = ((IBlockContainer) iTexture).getMeta();
if (meta == metaFromBlock && blockFromBlock == block) {
return (page << 7) + index;
}
}
}
}
}
throw new RuntimeException("Probably missing mapping or different texture class used for: "
+ blockFromBlock.getUnlocalizedName() + " meta:" + metaFromBlock);
}
/**
* Converts a Number to a String
*/
public static String parseNumberToString(int aNumber) {
boolean temp = true, negative = false;
if (aNumber < 0) {
aNumber *= -1;
negative = true;
}
StringBuilder tStringB = new StringBuilder();
for (int i = 1000000000; i > 0; i /= 10) {
int tDigit = (aNumber / i) % 10;
if (temp && tDigit != 0) temp = false;
if (!temp) {
tStringB.append(tDigit);
if (i != 1) for (int j = i; j > 0; j /= 1000) if (j == 1) tStringB.append(",");
}
}
String tString = tStringB.toString();
if (tString.equals(E)) tString = "0";
return negative ? "-" + tString : tString;
}
public static NBTTagCompound getNBTContainingBoolean(NBTTagCompound aNBT, Object aTag, boolean aValue) {
if (aNBT == null) aNBT = new NBTTagCompound();
aNBT.setBoolean(aTag.toString(), aValue);
return aNBT;
}
public static NBTTagCompound getNBTContainingByte(NBTTagCompound aNBT, Object aTag, byte aValue) {
if (aNBT == null) aNBT = new NBTTagCompound();
aNBT.setByte(aTag.toString(), aValue);
return aNBT;
}
public static NBTTagCompound getNBTContainingShort(NBTTagCompound aNBT, Object aTag, short aValue) {
if (aNBT == null) aNBT = new NBTTagCompound();
aNBT.setShort(aTag.toString(), aValue);
return aNBT;
}
public static NBTTagCompound getNBTContainingInteger(NBTTagCompound aNBT, Object aTag, int aValue) {
if (aNBT == null) aNBT = new NBTTagCompound();
aNBT.setInteger(aTag.toString(), aValue);
return aNBT;
}
public static NBTTagCompound getNBTContainingFloat(NBTTagCompound aNBT, Object aTag, float aValue) {
if (aNBT == null) aNBT = new NBTTagCompound();
aNBT.setFloat(aTag.toString(), aValue);
return aNBT;
}
public static NBTTagCompound getNBTContainingDouble(NBTTagCompound aNBT, Object aTag, double aValue) {
if (aNBT == null) aNBT = new NBTTagCompound();
aNBT.setDouble(aTag.toString(), aValue);
return aNBT;
}
public static NBTTagCompound getNBTContainingString(NBTTagCompound aNBT, Object aTag, Object aValue) {
if (aNBT == null) aNBT = new NBTTagCompound();
if (aValue == null) return aNBT;
aNBT.setString(aTag.toString(), aValue.toString());
return aNBT;
}
public static boolean isWearingFullFrostHazmat(EntityLivingBase aEntity) {
for (byte i = 1; i < 5; i++)
if (!isStackInList(aEntity.getEquipmentInSlot(i), GregTech_API.sFrostHazmatList)) return false;
return true;
}
public static boolean isWearingFullHeatHazmat(EntityLivingBase aEntity) {
for (byte i = 1; i < 5; i++)
if (!isStackInList(aEntity.getEquipmentInSlot(i), GregTech_API.sHeatHazmatList)) return false;
return true;
}
public static boolean isWearingFullBioHazmat(EntityLivingBase aEntity) {
for (byte i = 1; i < 5; i++)
if (!isStackInList(aEntity.getEquipmentInSlot(i), GregTech_API.sBioHazmatList)) return false;
return true;
}
public static boolean isWearingFullRadioHazmat(EntityLivingBase aEntity) {
for (byte i = 1; i < 5; i++)
if (!isStackInList(aEntity.getEquipmentInSlot(i), GregTech_API.sRadioHazmatList)) return false;
return true;
}
public static boolean isWearingFullElectroHazmat(EntityLivingBase aEntity) {
for (byte i = 1; i < 5; i++)
if (!isStackInList(aEntity.getEquipmentInSlot(i), GregTech_API.sElectroHazmatList)) return false;
return true;
}
public static boolean isWearingFullGasHazmat(EntityLivingBase aEntity) {
for (byte i = 1; i < 5; i++)
if (!isStackInList(aEntity.getEquipmentInSlot(i), GregTech_API.sGasHazmatList)) return false;
return true;
}
public static float getHeatDamageFromItem(ItemStack aStack) {
ItemData tData = GT_OreDictUnificator.getItemData(aStack);
return tData == null
? 0
: (tData.mPrefix == null ? 0 : tData.mPrefix.mHeatDamage)
+ (tData.hasValidMaterialData() ? tData.mMaterial.mMaterial.mHeatDamage : 0);
}
public static int getRadioactivityLevel(ItemStack aStack) {
ItemData tData = GT_OreDictUnificator.getItemData(aStack);
if (tData != null && tData.hasValidMaterialData()) {
if (tData.mMaterial.mMaterial.mEnchantmentArmors instanceof Enchantment_Radioactivity)
return tData.mMaterial.mMaterial.mEnchantmentArmorsLevel;
if (tData.mMaterial.mMaterial.mEnchantmentTools instanceof Enchantment_Radioactivity)
return tData.mMaterial.mMaterial.mEnchantmentToolsLevel;
}
return EnchantmentHelper.getEnchantmentLevel(Enchantment_Radioactivity.INSTANCE.effectId, aStack);
}
public static boolean isImmuneToBreathingGasses(EntityLivingBase aEntity) {
return isWearingFullGasHazmat(aEntity);
}
public static boolean applyHeatDamage(EntityLivingBase entity, float damage) {
return applyHeatDamage(entity, damage, GT_DamageSources.getHeatDamage());
}
public static boolean applyHeatDamageFromItem(EntityLivingBase entity, float damage, ItemStack item) {
return applyHeatDamage(entity, damage, new DamageSourceHotItem(item));
}
private static boolean applyHeatDamage(EntityLivingBase aEntity, float aDamage, DamageSource source) {
if (aDamage > 0
&& aEntity != null
&& aEntity.getActivePotionEffect(Potion.fireResistance) == null
&& !isWearingFullHeatHazmat(aEntity)) {
aEntity.attackEntityFrom(source, aDamage);
return true;
}
return false;
}
public static boolean applyFrostDamage(EntityLivingBase aEntity, float aDamage) {
if (aDamage > 0 && aEntity != null && !isWearingFullFrostHazmat(aEntity)) {
aEntity.attackEntityFrom(GT_DamageSources.getFrostDamage(), aDamage);
return true;
}
return false;
}
public static boolean applyElectricityDamage(EntityLivingBase aEntity, long aVoltage, long aAmperage) {
long aDamage = getTier(aVoltage) * aAmperage * 4;
if (aDamage > 0 && aEntity != null && !isWearingFullElectroHazmat(aEntity)) {
aEntity.attackEntityFrom(GT_DamageSources.getElectricDamage(), aDamage);
return true;
}
return false;
}
public static boolean applyRadioactivity(EntityLivingBase aEntity, int aLevel, int aAmountOfItems) {
if (aLevel > 0
&& aEntity != null
&& aEntity.getCreatureAttribute() != EnumCreatureAttribute.UNDEAD
&& aEntity.getCreatureAttribute() != EnumCreatureAttribute.ARTHROPOD
&& !isWearingFullRadioHazmat(aEntity)) {
PotionEffect tEffect = null;
aEntity.addPotionEffect(new PotionEffect(
Potion.moveSlowdown.id,
aLevel * 140 * aAmountOfItems
+ Math.max(
0,
((tEffect = aEntity.getActivePotionEffect(Potion.moveSlowdown)) == null
? 0
: tEffect.getDuration())),
Math.max(0, (5 * aLevel) / 7)));
aEntity.addPotionEffect(new PotionEffect(
Potion.digSlowdown.id,
aLevel * 150 * aAmountOfItems
+ Math.max(
0,
((tEffect = aEntity.getActivePotionEffect(Potion.digSlowdown)) == null
? 0
: tEffect.getDuration())),
Math.max(0, (5 * aLevel) / 7)));
aEntity.addPotionEffect(new PotionEffect(
Potion.confusion.id,
aLevel * 130 * aAmountOfItems
+ Math.max(
0,
((tEffect = aEntity.getActivePotionEffect(Potion.confusion)) == null
? 0
: tEffect.getDuration())),
Math.max(0, (5 * aLevel) / 7)));
aEntity.addPotionEffect(new PotionEffect(
Potion.weakness.id,
aLevel * 150 * aAmountOfItems
+ Math.max(
0,
((tEffect = aEntity.getActivePotionEffect(Potion.weakness)) == null
? 0
: tEffect.getDuration())),
Math.max(0, (5 * aLevel) / 7)));
aEntity.addPotionEffect(new PotionEffect(
Potion.hunger.id,
aLevel * 130 * aAmountOfItems
+ Math.max(
0,
((tEffect = aEntity.getActivePotionEffect(Potion.hunger)) == null
? 0
: tEffect.getDuration())),
Math.max(0, (5 * aLevel) / 7)));
aEntity.addPotionEffect(new PotionEffect(
24 /* IC2 Radiation */,
aLevel * 180 * aAmountOfItems
+ Math.max(
0,
((tEffect = aEntity.getActivePotionEffect(Potion.potionTypes[24])) == null
? 0
: tEffect.getDuration())),
Math.max(0, (5 * aLevel) / 7)));
return true;
}
return false;
}
public static ItemStack setStack(Object aSetStack, Object aToStack) {
if (isStackInvalid(aSetStack) || isStackInvalid(aToStack)) return null;
((ItemStack) aSetStack).func_150996_a(((ItemStack) aToStack).getItem());
((ItemStack) aSetStack).stackSize = ((ItemStack) aToStack).stackSize;
Items.feather.setDamage((ItemStack) aSetStack, Items.feather.getDamage((ItemStack) aToStack));
((ItemStack) aSetStack).setTagCompound(((ItemStack) aToStack).getTagCompound());
return (ItemStack) aSetStack;
}
public static FluidStack[] copyFluidArray(FluidStack... aStacks) {
FluidStack[] rStacks = new FluidStack[aStacks.length];
for (int i = 0; i < aStacks.length; i++) if (aStacks[i] != null) rStacks[i] = aStacks[i].copy();
return rStacks;
}
public static ItemStack[] copyStackArray(Object... aStacks) {
ItemStack[] rStacks = new ItemStack[aStacks.length];
for (int i = 0; i < aStacks.length; i++) rStacks[i] = copy(aStacks[i]);
return rStacks;
}
public static ItemStack copy(Object... aStacks) {
for (Object tStack : aStacks) if (isStackValid(tStack)) return ((ItemStack) tStack).copy();
return null;
}
@Nullable
public static ItemStack copyOrNull(@Nullable ItemStack stack) {
if (isStackValid(stack)) return stack.copy();
return null;
}
public static ItemStack copyAmount(long aAmount, Object... aStacks) {
ItemStack rStack = copy(aStacks);
if (isStackInvalid(rStack)) return null;
if (aAmount > 64) aAmount = 64;
else if (aAmount == -1) aAmount = 111;
else if (aAmount < 0) aAmount = 0;
rStack.stackSize = (byte) aAmount;
return rStack;
}
public static ItemStack copyAmountUnsafe(long aAmount, Object... aStacks) {
ItemStack rStack = copy(aStacks);
if (isStackInvalid(rStack)) return null;
if (aAmount > Integer.MAX_VALUE) aAmount = Integer.MAX_VALUE;
else if (aAmount < 0) aAmount = 0;
rStack.stackSize = (int) aAmount;
return rStack;
}
public static ItemStack copyMetaData(long aMetaData, Object... aStacks) {
ItemStack rStack = copy(aStacks);
if (isStackInvalid(rStack)) return null;
Items.feather.setDamage(rStack, (short) aMetaData);
return rStack;
}
public static ItemStack copyAmountAndMetaData(long aAmount, long aMetaData, Object... aStacks) {
ItemStack rStack = copyAmount(aAmount, aStacks);
if (isStackInvalid(rStack)) return null;
Items.feather.setDamage(rStack, (short) aMetaData);
return rStack;
}
/**
* returns a copy of an ItemStack with its Stacksize being multiplied by aMultiplier
*/
public static ItemStack mul(long aMultiplier, Object... aStacks) {
ItemStack rStack = copy(aStacks);
if (rStack == null) return null;
rStack.stackSize *= aMultiplier;
return rStack;
}
/**
* Loads an ItemStack properly.
*/
public static ItemStack loadItem(NBTTagCompound aNBT, String aTagName) {
return loadItem(aNBT.getCompoundTag(aTagName));
}
public static FluidStack loadFluid(NBTTagCompound aNBT, String aTagName) {
return loadFluid(aNBT.getCompoundTag(aTagName));
}
/**
* Loads an ItemStack properly.
*/
public static ItemStack loadItem(NBTTagCompound aNBT) {
if (aNBT == null) return null;
return GT_OreDictUnificator.get(true, ItemStack.loadItemStackFromNBT(aNBT));
}
/**
* Loads an FluidStack properly.
*/
public static FluidStack loadFluid(NBTTagCompound aNBT) {
if (aNBT == null) return null;
return FluidStack.loadFluidStackFromNBT(aNBT);
}
public static E selectItemInList(int aIndex, E aReplacement, List aList) {
if (aList == null || aList.isEmpty()) return aReplacement;
if (aList.size() <= aIndex) return aList.get(aList.size() - 1);
if (aIndex < 0) return aList.get(0);
return aList.get(aIndex);
}
public static E selectItemInList(int aIndex, E aReplacement, E... aList) {
if (aList == null || aList.length == 0) return aReplacement;
if (aList.length <= aIndex) return aList[aList.length - 1];
if (aIndex < 0) return aList[0];
return aList[aIndex];
}
public static boolean isStackInList(ItemStack aStack, Collection aList) {
if (aStack == null) {
return false;
}
return isStackInList(new GT_ItemStack(aStack), aList);
}
public static boolean isStackInList(ItemStack aStack, Set aList) {
if (aStack == null) {
return false;
}
return isStackInList(new GT_ItemStack2(aStack), aList);
}
public static boolean isStackInList(GT_ItemStack aStack, Collection aList) {
return aStack != null
&& (aList.contains(aStack) || aList.contains(new GT_ItemStack(aStack.mItem, aStack.mStackSize, W)));
}
public static boolean isStackInList(GT_ItemStack2 aStack, Set aList) {
return aStack != null
&& (aList.contains(aStack) || aList.contains(new GT_ItemStack2(aStack.mItem, aStack.mStackSize, W)));
}
/**
* re-maps all Keys of a Map after the Keys were weakened.
*/
public static Map reMap(Map aMap) {
Map tMap = new HashMap<>(aMap);
aMap.clear();
aMap.putAll(tMap);
return aMap;
}
public static > LinkedHashMap sortMapByValuesAcending(Map map) {
return map.entrySet().stream()
.sorted(Entry.comparingByValue())
.collect(CollectorUtils.entriesToMap(LinkedHashMap::new));
}
/**
* Why the fuck do neither Java nor Guava have a Function to do this?
*/
public static LinkedHashMap sortMapByValuesDescending(Map aMap) {
List> tEntrySet = new LinkedList<>(aMap.entrySet());
tEntrySet.sort((aValue1, aValue2) -> {
return aValue2.getValue().compareTo(aValue1.getValue()); // FB: RV - RV_NEGATING_RESULT_OF_COMPARETO
});
LinkedHashMap rMap = new LinkedHashMap<>();
for (Map.Entry tEntry : tEntrySet) rMap.put(tEntry.getKey(), tEntry.getValue());
return rMap;
}
/**
* Translates a Material Amount into an Amount of Fluid in Fluid Material Units.
*/
public static long translateMaterialToFluidAmount(long aMaterialAmount, boolean aRoundUp) {
return translateMaterialToAmount(aMaterialAmount, L, aRoundUp);
}
/**
* Translates a Material Amount into an Amount of Fluid. Second Parameter for things like Bucket Amounts (1000) and similar
*/
public static long translateMaterialToAmount(long aMaterialAmount, long aAmountPerUnit, boolean aRoundUp) {
return Math.max(
0,
((aMaterialAmount * aAmountPerUnit) / M)
+ (aRoundUp && (aMaterialAmount * aAmountPerUnit) % M > 0 ? 1 : 0));
}
/**
* This checks if the Dimension is really a Dimension and not another Planet or something.
* Used for my Teleporter.
*/
public static boolean isRealDimension(int aDimensionID) {
if (aDimensionID <= 1 && aDimensionID >= -1 && !GregTech_API.sDimensionalList.contains(aDimensionID))
return true;
return !GregTech_API.sDimensionalList.contains(aDimensionID)
&& DimensionManager.isDimensionRegistered(aDimensionID);
}
public static boolean moveEntityToDimensionAtCoords(
Entity entity, int aDimension, double aX, double aY, double aZ) {
// Credit goes to BrandonCore Author :!:
if (entity == null || entity.worldObj.isRemote) return false;
if (entity.ridingEntity != null) entity.mountEntity(null);
if (entity.riddenByEntity != null) entity.riddenByEntity.mountEntity(null);
World startWorld = entity.worldObj;
World destinationWorld =
FMLCommonHandler.instance().getMinecraftServerInstance().worldServerForDimension(aDimension);
if (destinationWorld == null) {
return false;
}
boolean interDimensional = startWorld.provider.dimensionId != destinationWorld.provider.dimensionId;
if (!interDimensional) return false;
startWorld.updateEntityWithOptionalForce(entity, false); // added
if ((entity instanceof EntityPlayerMP) && interDimensional) {
EntityPlayerMP player = (EntityPlayerMP) entity;
player.closeScreen(); // added
player.dimension = aDimension;
player.playerNetServerHandler.sendPacket(new S07PacketRespawn(
player.dimension,
player.worldObj.difficultySetting,
destinationWorld.getWorldInfo().getTerrainType(),
player.theItemInWorldManager.getGameType()));
((WorldServer) startWorld).getPlayerManager().removePlayer(player);
startWorld.playerEntities.remove(player);
startWorld.updateAllPlayersSleepingFlag();
int i = entity.chunkCoordX;
int j = entity.chunkCoordZ;
if ((entity.addedToChunk) && (startWorld.getChunkProvider().chunkExists(i, j))) {
startWorld.getChunkFromChunkCoords(i, j).removeEntity(entity);
startWorld.getChunkFromChunkCoords(i, j).isModified = true;
}
startWorld.loadedEntityList.remove(entity);
startWorld.onEntityRemoved(entity);
}
entity.setLocationAndAngles(aX, aY, aZ, entity.rotationYaw, entity.rotationPitch);
((WorldServer) destinationWorld).theChunkProviderServer.loadChunk((int) aX >> 4, (int) aZ >> 4);
destinationWorld.theProfiler.startSection("placing");
if (interDimensional) {
if (!(entity instanceof EntityPlayer)) {
NBTTagCompound entityNBT = new NBTTagCompound();
entity.isDead = false;
entityNBT.setString("id", EntityList.getEntityString(entity));
entity.writeToNBT(entityNBT);
entity.isDead = true;
entity = EntityList.createEntityFromNBT(entityNBT, destinationWorld);
if (entity == null) {
return false;
}
entity.dimension = destinationWorld.provider.dimensionId;
}
destinationWorld.spawnEntityInWorld(entity);
entity.setWorld(destinationWorld);
}
entity.setLocationAndAngles(aX, aY, aZ, entity.rotationYaw, entity.rotationPitch);
destinationWorld.updateEntityWithOptionalForce(entity, false);
entity.setLocationAndAngles(aX, aY, aZ, entity.rotationYaw, entity.rotationPitch);
if ((entity instanceof EntityPlayerMP)) {
EntityPlayerMP player = (EntityPlayerMP) entity;
if (interDimensional) {
player.mcServer.getConfigurationManager().func_72375_a(player, (WorldServer) destinationWorld);
}
player.playerNetServerHandler.setPlayerLocation(aX, aY, aZ, player.rotationYaw, player.rotationPitch);
}
destinationWorld.updateEntityWithOptionalForce(entity, false);
if (((entity instanceof EntityPlayerMP)) && interDimensional) {
EntityPlayerMP player = (EntityPlayerMP) entity;
player.theItemInWorldManager.setWorld((WorldServer) destinationWorld);
player.mcServer.getConfigurationManager().updateTimeAndWeatherForPlayer(player, (WorldServer)
destinationWorld);
player.mcServer.getConfigurationManager().syncPlayerInventory(player);
for (PotionEffect potionEffect : (Iterable) player.getActivePotionEffects()) {
player.playerNetServerHandler.sendPacket(new S1DPacketEntityEffect(player.getEntityId(), potionEffect));
}
player.playerNetServerHandler.sendPacket(
new S1FPacketSetExperience(player.experience, player.experienceTotal, player.experienceLevel));
FMLCommonHandler.instance()
.firePlayerChangedDimensionEvent(
player, startWorld.provider.dimensionId, destinationWorld.provider.dimensionId);
}
entity.setLocationAndAngles(aX, aY, aZ, entity.rotationYaw, entity.rotationPitch);
destinationWorld.theProfiler.endSection();
entity.fallDistance = 0;
return true;
}
public static int getScaleCoordinates(double aValue, int aScale) {
return (int) Math.floor(aValue / aScale);
}
public static int getCoordinateScan(
ArrayList aList,
EntityPlayer aPlayer,
World aWorld,
int aScanLevel,
int aX,
int aY,
int aZ,
int aSide,
float aClickX,
float aClickY,
float aClickZ) {
if (aList == null) return 0;
ArrayList tList = new ArrayList<>();
int rEUAmount = 0;
TileEntity tTileEntity = aWorld.getTileEntity(aX, aY, aZ);
final Block tBlock = aWorld.getBlock(aX, aY, aZ);
tList.add("----- X: " + EnumChatFormatting.AQUA + formatNumbers(aX) + EnumChatFormatting.RESET + " Y: "
+ EnumChatFormatting.AQUA + formatNumbers(aY) + EnumChatFormatting.RESET + " Z: "
+ EnumChatFormatting.AQUA + formatNumbers(aZ) + EnumChatFormatting.RESET + " D: "
+ EnumChatFormatting.AQUA + aWorld.provider.dimensionId + EnumChatFormatting.RESET + " -----");
try {
if (tTileEntity instanceof IInventory)
tList.add(GT_Utility.trans("162", "Name: ")
+ EnumChatFormatting.BLUE
+ ((IInventory) tTileEntity).getInventoryName()
+ EnumChatFormatting.RESET
+ GT_Utility.trans("163", " MetaData: ")
+ EnumChatFormatting.AQUA
+ aWorld.getBlockMetadata(aX, aY, aZ)
+ EnumChatFormatting.RESET);
else
tList.add(GT_Utility.trans("162", "Name: ")
+ EnumChatFormatting.BLUE
+ tBlock.getUnlocalizedName()
+ EnumChatFormatting.RESET
+ GT_Utility.trans("163", " MetaData: ")
+ EnumChatFormatting.AQUA
+ aWorld.getBlockMetadata(aX, aY, aZ)
+ EnumChatFormatting.RESET);
tList.add(GT_Utility.trans("164", "Hardness: ")
+ EnumChatFormatting.YELLOW
+ tBlock.getBlockHardness(aWorld, aX, aY, aZ)
+ EnumChatFormatting.RESET
+ GT_Utility.trans("165", " Blast Resistance: ")
+ EnumChatFormatting.YELLOW
+ tBlock.getExplosionResistance(
aPlayer, aWorld, aX, aY, aZ, aPlayer.posX, aPlayer.posY, aPlayer.posZ)
+ EnumChatFormatting.RESET);
if (tBlock.isBeaconBase(aWorld, aX, aY, aZ, aX, aY + 1, aZ))
tList.add(EnumChatFormatting.GOLD
+ GT_Utility.trans("166", "Is valid Beacon Pyramid Material")
+ EnumChatFormatting.RESET);
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
if (tTileEntity != null) {
try {
if (tTileEntity instanceof IFluidHandler) {
rEUAmount += 500;
final FluidTankInfo[] tTanks =
((IFluidHandler) tTileEntity).getTankInfo(ForgeDirection.getOrientation(aSide));
if (tTanks != null)
for (byte i = 0; i < tTanks.length; i++) {
tList.add(GT_Utility.trans("167", "Tank ") + i + ": " + EnumChatFormatting.GREEN
+ formatNumbers((tTanks[i].fluid == null ? 0 : tTanks[i].fluid.amount))
+ EnumChatFormatting.RESET + " L / " + EnumChatFormatting.YELLOW
+ formatNumbers(tTanks[i].capacity) + EnumChatFormatting.RESET + " L "
+ EnumChatFormatting.GOLD + getFluidName(tTanks[i].fluid, true)
+ EnumChatFormatting.RESET);
}
}
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
try {
if (tTileEntity instanceof ic2.api.reactor.IReactorChamber) {
rEUAmount += 500;
tTileEntity = (TileEntity) (((ic2.api.reactor.IReactorChamber) tTileEntity).getReactor());
}
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
try {
if (tTileEntity instanceof ic2.api.reactor.IReactor) {
rEUAmount += 500;
tList.add(GT_Utility.trans("168", "Heat: ") + EnumChatFormatting.GREEN
+ formatNumbers(((ic2.api.reactor.IReactor) tTileEntity).getHeat())
+ EnumChatFormatting.RESET + " / " + EnumChatFormatting.YELLOW
+ formatNumbers(((ic2.api.reactor.IReactor) tTileEntity).getMaxHeat())
+ EnumChatFormatting.RESET);
tList.add(
GT_Utility.trans("169", "HEM: ")
+ EnumChatFormatting.YELLOW
+ ((ic2.api.reactor.IReactor) tTileEntity).getHeatEffectModifier()
+ EnumChatFormatting.RESET
/* + GT_Utility.trans("170"," Base EU Output: ")/* + ((ic2.api.reactor.IReactor)tTileEntity).getOutput()*/ );
}
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
try {
if (tTileEntity instanceof IAlignmentProvider) {
final IAlignment tAlignment = ((IAlignmentProvider) tTileEntity).getAlignment();
if (tAlignment != null) {
rEUAmount += 100;
tList.add(GT_Utility.trans("219", "Extended Facing: ")
+ EnumChatFormatting.GREEN
+ tAlignment.getExtendedFacing()
+ EnumChatFormatting.RESET);
}
}
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
try {
if (tTileEntity instanceof ic2.api.tile.IWrenchable) {
rEUAmount += 100;
tList.add(GT_Utility.trans("171", "Facing: ") + EnumChatFormatting.GREEN
+ ((ic2.api.tile.IWrenchable) tTileEntity).getFacing() + EnumChatFormatting.RESET
+ GT_Utility.trans("172", " / Chance: ") + EnumChatFormatting.YELLOW
+ (((ic2.api.tile.IWrenchable) tTileEntity).getWrenchDropRate() * 100)
+ EnumChatFormatting.RESET + "%");
tList.add(
((ic2.api.tile.IWrenchable) tTileEntity).wrenchCanRemove(aPlayer)
? EnumChatFormatting.GREEN
+ GT_Utility.trans("173", "You can remove this with a Wrench")
+ EnumChatFormatting.RESET
: EnumChatFormatting.RED
+ GT_Utility.trans("174", "You can NOT remove this with a Wrench")
+ EnumChatFormatting.RESET);
}
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
try {
if (tTileEntity instanceof ic2.api.energy.tile.IEnergyConductor) {
rEUAmount += 200;
tList.add(GT_Utility.trans("175", "Conduction Loss: ")
+ EnumChatFormatting.YELLOW
+ ((ic2.api.energy.tile.IEnergyConductor) tTileEntity).getConductionLoss()
+ EnumChatFormatting.RESET);
}
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
try {
if (tTileEntity instanceof ic2.api.tile.IEnergyStorage) {
rEUAmount += 200;
tList.add(GT_Utility.trans("176", "Contained Energy: ") + EnumChatFormatting.YELLOW
+ formatNumbers(((ic2.api.tile.IEnergyStorage) tTileEntity).getStored())
+ EnumChatFormatting.RESET + " EU / " + EnumChatFormatting.YELLOW
+ formatNumbers(((ic2.api.tile.IEnergyStorage) tTileEntity).getCapacity())
+ EnumChatFormatting.RESET + " EU");
}
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
try {
if (tTileEntity instanceof IUpgradableMachine) {
rEUAmount += 500;
if (((IUpgradableMachine) tTileEntity).hasMufflerUpgrade())
tList.add(EnumChatFormatting.GREEN
+ GT_Utility.trans("177", "Has Muffler Upgrade")
+ EnumChatFormatting.RESET);
}
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
try {
if (tTileEntity instanceof IMachineProgress) {
if (((IMachineProgress) tTileEntity).isAllowedToWork()) {
tList.add(EnumChatFormatting.RED + "Disabled." + EnumChatFormatting.RESET);
}
if (((IMachineProgress) tTileEntity).wasShutdown()) {
tList.add(EnumChatFormatting.RED + "Shut down due to power loss." + EnumChatFormatting.RESET);
}
rEUAmount += 400;
int tValue = 0;
if (0 < (tValue = ((IMachineProgress) tTileEntity).getMaxProgress()))
tList.add(GT_Utility.trans("178", "Progress/Load: ") + EnumChatFormatting.GREEN
+ formatNumbers(((IMachineProgress) tTileEntity).getProgress())
+ EnumChatFormatting.RESET + " / " + EnumChatFormatting.YELLOW + formatNumbers(tValue)
+ EnumChatFormatting.RESET);
}
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
try {
if (tTileEntity instanceof ICoverable) {
rEUAmount += 300;
final String tString = ((ICoverable) tTileEntity)
.getCoverBehaviorAtSideNew((byte) aSide)
.getDescription(
(byte) aSide,
((ICoverable) tTileEntity).getCoverIDAtSide((byte) aSide),
((ICoverable) tTileEntity).getComplexCoverDataAtSide((byte) aSide),
(ICoverable) tTileEntity);
if (tString != null && !tString.equals(E)) tList.add(tString);
}
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
try {
if (tTileEntity instanceof IBasicEnergyContainer
&& ((IBasicEnergyContainer) tTileEntity).getEUCapacity() > 0) {
tList.add(GT_Utility.trans("179", "Max IN: ") + EnumChatFormatting.RED
+ formatNumbers(((IBasicEnergyContainer) tTileEntity).getInputVoltage()) + " ("
+ GT_Values.VN[getTier(((IBasicEnergyContainer) tTileEntity).getInputVoltage())] + ") "
+ EnumChatFormatting.RESET + GT_Utility.trans("182", " EU at ") + EnumChatFormatting.RED
+ formatNumbers(((IBasicEnergyContainer) tTileEntity).getInputAmperage())
+ EnumChatFormatting.RESET + GT_Utility.trans("183", " A"));
tList.add(GT_Utility.trans("181", "Max OUT: ") + EnumChatFormatting.RED
+ formatNumbers(((IBasicEnergyContainer) tTileEntity).getOutputVoltage()) + " ("
+ GT_Values.VN[getTier(((IBasicEnergyContainer) tTileEntity).getOutputVoltage())] + ") "
+ EnumChatFormatting.RESET + GT_Utility.trans("182", " EU at ") + EnumChatFormatting.RED
+ formatNumbers(((IBasicEnergyContainer) tTileEntity).getOutputAmperage())
+ EnumChatFormatting.RESET + GT_Utility.trans("183", " A"));
tList.add(GT_Utility.trans("184", "Energy: ") + EnumChatFormatting.GREEN
+ formatNumbers(((IBasicEnergyContainer) tTileEntity).getStoredEU())
+ EnumChatFormatting.RESET + " EU / " + EnumChatFormatting.YELLOW
+ formatNumbers(((IBasicEnergyContainer) tTileEntity).getEUCapacity())
+ EnumChatFormatting.RESET + " EU");
}
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
try {
if (tTileEntity instanceof IGregTechTileEntity) {
tList.add(GT_Utility.trans("186", "Owned by: ")
+ EnumChatFormatting.BLUE
+ ((IGregTechTileEntity) tTileEntity).getOwnerName()
+ EnumChatFormatting.RESET);
}
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
try {
if (tTileEntity instanceof IGregTechDeviceInformation
&& ((IGregTechDeviceInformation) tTileEntity).isGivingInformation()) {
tList.addAll(Arrays.asList(((IGregTechDeviceInformation) tTileEntity).getInfoData()));
}
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
try {
if (tTileEntity instanceof ic2.api.crops.ICropTile) {
rEUAmount += 1000;
final ic2.api.crops.ICropTile crop = (ic2.api.crops.ICropTile) tTileEntity;
if (crop.getScanLevel() < 4) crop.setScanLevel((byte) 4);
if (crop.getCrop() != null) {
tList.add(GT_Utility.trans("187", "Type -- Crop-Name: ")
+ crop.getCrop().name()
+ GT_Utility.trans("188", " Growth: ")
+ crop.getGrowth()
+ GT_Utility.trans("189", " Gain: ")
+ crop.getGain()
+ GT_Utility.trans("190", " Resistance: ")
+ crop.getResistance());
}
tList.add(GT_Utility.trans("191", "Plant -- Fertilizer: ")
+ crop.getNutrientStorage()
+ GT_Utility.trans("192", " Water: ")
+ crop.getHydrationStorage()
+ GT_Utility.trans("193", " Weed-Ex: ")
+ crop.getWeedExStorage()
+ GT_Utility.trans("194", " Scan-Level: ")
+ crop.getScanLevel());
tList.add(GT_Utility.trans("195", "Environment -- Nutrients: ")
+ crop.getNutrients()
+ GT_Utility.trans("196", " Humidity: ")
+ crop.getHumidity()
+ GT_Utility.trans("197", " Air-Quality: ")
+ crop.getAirQuality());
if (crop.getCrop() != null) {
final StringBuilder tStringB = new StringBuilder();
for (String tAttribute : crop.getCrop().attributes()) {
tStringB.append(", ").append(tAttribute);
}
final String tString = tStringB.toString();
tList.add(GT_Utility.trans("198", "Attributes:") + tString.replaceFirst(",", E));
tList.add(GT_Utility.trans("199", "Discovered by: ")
+ crop.getCrop().discoveredBy());
}
}
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
try {
if (tTileEntity instanceof forestry.arboriculture.tiles.TileLeaves) {
final forestry.arboriculture.tiles.TileLeaves tileLeaves =
(forestry.arboriculture.tiles.TileLeaves) tTileEntity;
final forestry.api.arboriculture.ITree tree = tileLeaves.getTree();
if (tree != null) {
rEUAmount += 1000;
if (!tree.isAnalyzed()) tree.analyze();
tree.addTooltip(tList);
}
}
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
}
final Chunk currentChunk = aWorld.getChunkFromBlockCoords(aX, aZ);
if (aPlayer.capabilities.isCreativeMode) {
final FluidStack tFluid = undergroundOilReadInformation(currentChunk); // -# to only read
if (tFluid != null)
tList.add(EnumChatFormatting.GOLD + tFluid.getLocalizedName() + EnumChatFormatting.RESET + ": "
+ EnumChatFormatting.YELLOW + formatNumbers(tFluid.amount) + EnumChatFormatting.RESET + " L");
else
tList.add(EnumChatFormatting.GOLD + GT_Utility.trans("201", "Nothing") + EnumChatFormatting.RESET + ": "
+ EnumChatFormatting.YELLOW + '0' + EnumChatFormatting.RESET + " L");
}
// if(aPlayer.capabilities.isCreativeMode){
if (GT_Pollution.hasPollution(currentChunk)) {
tList.add(GT_Utility.trans("202", "Pollution in Chunk: ")
+ EnumChatFormatting.RED
+ formatNumbers(GT_Pollution.getPollution(currentChunk))
+ EnumChatFormatting.RESET
+ GT_Utility.trans("203", " gibbl"));
} else {
tList.add(EnumChatFormatting.GREEN
+ GT_Utility.trans("204", "No Pollution in Chunk! HAYO!")
+ EnumChatFormatting.RESET);
}
try {
if (tBlock instanceof IDebugableBlock) {
rEUAmount += 500;
final ArrayList temp = ((IDebugableBlock) tBlock).getDebugInfo(aPlayer, aX, aY, aZ, 3);
if (temp != null) tList.addAll(temp);
}
} catch (Throwable e) {
if (D1) e.printStackTrace(GT_Log.err);
}
final BlockScanningEvent tEvent = new BlockScanningEvent(
aWorld,
aPlayer,
aX,
aY,
aZ,
(byte) aSide,
aScanLevel,
tBlock,
tTileEntity,
tList,
aClickX,
aClickY,
aClickZ);
tEvent.mEUCost = rEUAmount;
MinecraftForge.EVENT_BUS.post(tEvent);
if (!tEvent.isCanceled()) aList.addAll(tList);
return tEvent.mEUCost;
}
public static String trans(String aKey, String aEnglish) {
return GT_LanguageManager.addStringLocalization("Interaction_DESCRIPTION_Index_" + aKey, aEnglish, false);
}
public static String getTrans(String aKey) {
return GT_LanguageManager.getTranslation("Interaction_DESCRIPTION_Index_" + aKey);
}
/**
* @return an Array containing the X and the Y Coordinate of the clicked Point, with the top left Corner as Origin, like on the Texture Sheet. return values should always be between [0.0F and 0.99F].
*/
public static float[] getClickedFacingCoords(byte aSide, float aX, float aY, float aZ) {
switch (aSide) {
case 0:
return new float[] {Math.min(0.99F, Math.max(0, 1 - aX)), Math.min(0.99F, Math.max(0, aZ))};
case 1:
return new float[] {Math.min(0.99F, Math.max(0, aX)), Math.min(0.99F, Math.max(0, aZ))};
case 2:
return new float[] {Math.min(0.99F, Math.max(0, 1 - aX)), Math.min(0.99F, Math.max(0, 1 - aY))};
case 3:
return new float[] {Math.min(0.99F, Math.max(0, aX)), Math.min(0.99F, Math.max(0, 1 - aY))};
case 4:
return new float[] {Math.min(0.99F, Math.max(0, aZ)), Math.min(0.99F, Math.max(0, 1 - aY))};
case 5:
return new float[] {Math.min(0.99F, Math.max(0, 1 - aZ)), Math.min(0.99F, Math.max(0, 1 - aY))};
default:
return new float[] {0.5F, 0.5F};
}
}
/**
* This Function determines the direction a Block gets when being Wrenched.
* returns -1 if invalid. Even though that could never happen.
*/
public static byte determineWrenchingSide(byte aSide, float aX, float aY, float aZ) {
byte tBack = getOppositeSide(aSide);
switch (aSide) {
case 0:
case 1:
if (aX < 0.25) {
if (aZ < 0.25) return tBack;
if (aZ > 0.75) return tBack;
return 4;
}
if (aX > 0.75) {
if (aZ < 0.25) return tBack;
if (aZ > 0.75) return tBack;
return 5;
}
if (aZ < 0.25) return 2;
if (aZ > 0.75) return 3;
return aSide;
case 2:
case 3:
if (aX < 0.25) {
if (aY < 0.25) return tBack;
if (aY > 0.75) return tBack;
return 4;
}
if (aX > 0.75) {
if (aY < 0.25) return tBack;
if (aY > 0.75) return tBack;
return 5;
}
if (aY < 0.25) return 0;
if (aY > 0.75) return 1;
return aSide;
case 4:
case 5:
if (aZ < 0.25) {
if (aY < 0.25) return tBack;
if (aY > 0.75) return tBack;
return 2;
}
if (aZ > 0.75) {
if (aY < 0.25) return tBack;
if (aY > 0.75) return tBack;
return 3;
}
if (aY < 0.25) return 0;
if (aY > 0.75) return 1;
return aSide;
}
return -1;
}
private static DecimalFormat getDecimalFormat() {
return decimalFormatters.computeIfAbsent(Locale.getDefault(Locale.Category.FORMAT), locale -> {
DecimalFormat numberFormat = new DecimalFormat(); // uses the necessary locale inside anyway
numberFormat.setGroupingUsed(true);
numberFormat.setMaximumFractionDigits(2);
numberFormat.setRoundingMode(RoundingMode.HALF_UP);
DecimalFormatSymbols decimalFormatSymbols = numberFormat.getDecimalFormatSymbols();
decimalFormatSymbols.setGroupingSeparator(','); // Use sensible separator for best clarity.
numberFormat.setDecimalFormatSymbols(decimalFormatSymbols);
return numberFormat;
});
}
public static String formatNumbers(BigInteger aNumber) {
return getDecimalFormat().format(aNumber);
}
public static String formatNumbers(long aNumber) {
return getDecimalFormat().format(aNumber);
}
public static String formatNumbers(double aNumber) {
return getDecimalFormat().format(aNumber);
}
/*
* Check if stack has enough items of given type and subtract from stack, if there's no creative or 111 stack.
*/
public static boolean consumeItems(EntityPlayer player, ItemStack stack, Item item, int count) {
if (stack != null && stack.getItem() == item && stack.stackSize >= count) {
if ((!player.capabilities.isCreativeMode) && (stack.stackSize != 111)) stack.stackSize -= count;
return true;
}
return false;
}
/*
* Check if stack has enough items of given gregtech material (will be oredicted)
* and subtract from stack, if there's no creative or 111 stack.
*/
public static boolean consumeItems(
EntityPlayer player, ItemStack stack, gregtech.api.enums.Materials mat, int count) {
if (stack != null
&& GT_OreDictUnificator.getItemData(stack).mMaterial.mMaterial == mat
&& stack.stackSize >= count) {
if ((!player.capabilities.isCreativeMode) && (stack.stackSize != 111)) stack.stackSize -= count;
return true;
}
return false;
}
public static ArrayList sortByValueToList(Map map) {
List> list = new LinkedList<>(map.entrySet());
list.sort((o1, o2) -> o2.getValue() - o1.getValue());
ArrayList result = new ArrayList<>();
for (Map.Entry e : list) result.add(e.getKey());
return result;
}
public static String joinListToString(List list) {
StringBuilder result = new StringBuilder(32);
for (String s : list) result.append(result.length() == 0 ? s : '|' + s);
return result.toString();
}
public static ItemStack getIntegratedCircuit(int config) {
return ItemList.Circuit_Integrated.getWithDamage(0, config);
}
public static float getBlockHardnessAt(World aWorld, int aX, int aY, int aZ) {
return aWorld.getBlock(aX, aY, aZ).getBlockHardness(aWorld, aX, aY, aZ);
}
public static FakePlayer getFakePlayer(IGregTechTileEntity aBaseMetaTileEntity) {
if (aBaseMetaTileEntity.getWorld() instanceof WorldServer) {
return FakePlayerFactory.get(
(WorldServer) aBaseMetaTileEntity.getWorld(),
new GameProfile(aBaseMetaTileEntity.getOwnerUuid(), aBaseMetaTileEntity.getOwnerName()));
}
return null;
}
public static boolean eraseBlockByFakePlayer(FakePlayer aPlayer, int aX, int aY, int aZ, boolean isSimulate) {
if (aPlayer == null) return false;
World aWorld = aPlayer.worldObj;
BlockEvent.BreakEvent event = new BlockEvent.BreakEvent(
aX, aY, aZ, aWorld, aWorld.getBlock(aX, aY, aZ), aWorld.getBlockMetadata(aX, aY, aZ), aPlayer);
MinecraftForge.EVENT_BUS.post(event);
if (!event.isCanceled()) {
if (!isSimulate) return aWorld.setBlockToAir(aX, aY, aZ);
return true;
}
return false;
}
public static boolean setBlockByFakePlayer(
FakePlayer aPlayer, int aX, int aY, int aZ, Block aBlock, int aMeta, boolean isSimulate) {
if (aPlayer == null) return false;
World aWorld = aPlayer.worldObj;
BlockEvent.PlaceEvent event = ForgeEventFactory.onPlayerBlockPlace(
aPlayer, new BlockSnapshot(aWorld, aX, aY, aZ, aBlock, aMeta), ForgeDirection.UNKNOWN);
if (!event.isCanceled()) {
if (!isSimulate) return aWorld.setBlock(aX, aY, aZ, aBlock, aMeta, 3);
return true;
}
return false;
}
public static int findMatchingStackInList(List aStacks, ItemStack aStack) {
if (isStackInvalid(aStack)) return -1;
for (int i = 0, aStacksSize = aStacks.size(); i < aStacksSize; i++) {
ItemStack tStack = aStacks.get(i);
if (areStacksEqual(aStack, tStack)) return i;
}
return -1;
}
public static class ItemNBT {
public static void setNBT(ItemStack aStack, NBTTagCompound aNBT) {
if (aNBT == null) {
aStack.setTagCompound(null);
return;
}
ArrayList tTagsToRemove = new ArrayList<>();
for (Object tKey : aNBT.func_150296_c()) {
NBTBase tValue = aNBT.getTag((String) tKey);
if (tValue == null
|| (tValue instanceof NBTPrimitive && ((NBTPrimitive) tValue).func_150291_c() == 0)
|| (tValue instanceof NBTTagString
&& isStringInvalid(((NBTTagString) tValue).func_150285_a_())))
tTagsToRemove.add((String) tKey);
}
for (Object tKey : tTagsToRemove) aNBT.removeTag((String) tKey);
aStack.setTagCompound(aNBT.hasNoTags() ? null : aNBT);
}
public static NBTTagCompound getNBT(ItemStack aStack) {
NBTTagCompound rNBT = aStack.getTagCompound();
return rNBT == null ? new NBTTagCompound() : rNBT;
}
public static void setPunchCardData(ItemStack aStack, String aPunchCardData) {
NBTTagCompound tNBT = getNBT(aStack);
tNBT.setString("GT.PunchCardData", aPunchCardData);
setNBT(aStack, tNBT);
}
public static String getPunchCardData(ItemStack aStack) {
NBTTagCompound tNBT = getNBT(aStack);
return tNBT.getString("GT.PunchCardData");
}
public static void setLighterFuel(ItemStack aStack, long aFuel) {
NBTTagCompound tNBT = getNBT(aStack);
tNBT.setLong("GT.LighterFuel", aFuel);
setNBT(aStack, tNBT);
}
public static long getLighterFuel(ItemStack aStack) {
NBTTagCompound tNBT = getNBT(aStack);
return tNBT.getLong("GT.LighterFuel");
}
public static void setMapID(ItemStack aStack, short aMapID) {
NBTTagCompound tNBT = getNBT(aStack);
tNBT.setShort("map_id", aMapID);
setNBT(aStack, tNBT);
}
public static short getMapID(ItemStack aStack) {
NBTTagCompound tNBT = getNBT(aStack);
if (!tNBT.hasKey("map_id")) return -1;
return tNBT.getShort("map_id");
}
public static void setBookTitle(ItemStack aStack, String aTitle) {
NBTTagCompound tNBT = getNBT(aStack);
tNBT.setString("title", aTitle);
setNBT(aStack, tNBT);
}
public static String getBookTitle(ItemStack aStack) {
NBTTagCompound tNBT = getNBT(aStack);
return tNBT.getString("title");
}
public static void setBookAuthor(ItemStack aStack, String aAuthor) {
NBTTagCompound tNBT = getNBT(aStack);
tNBT.setString("author", aAuthor);
setNBT(aStack, tNBT);
}
public static String getBookAuthor(ItemStack aStack) {
NBTTagCompound tNBT = getNBT(aStack);
return tNBT.getString("author");
}
public static void setProspectionData(
ItemStack aStack, int aX, int aY, int aZ, int aDim, FluidStack aFluid, String... aOres) {
NBTTagCompound tNBT = getNBT(aStack);
StringBuilder tData = new StringBuilder(aX + "," + aY + "," + aZ + "," + aDim + ",");
if (aFluid != null)
tData.append(aFluid.amount)
.append(",")
.append(aFluid.getLocalizedName())
.append(","); // TODO CHECK IF THAT /5000 is needed (Not needed)
for (String tString : aOres) {
tData.append(tString).append(",");
}
tNBT.setString("prospection", tData.toString());
setNBT(aStack, tNBT);
}
public static void setAdvancedProspectionData(
byte aTier,
ItemStack aStack,
int aX,
short aY,
int aZ,
int aDim,
ArrayList aOils,
ArrayList aOres,
int aRadius) {
setBookTitle(aStack, "Raw Prospection Data");
NBTTagCompound tNBT = getNBT(aStack);
tNBT.setByte("prospection_tier", aTier);
tNBT.setString("prospection_pos", "Dim: " + aDim + "\nX: " + aX + " Y: " + aY + " Z: " + aZ);
// ores
Collections.sort(aOres);
tNBT.setString("prospection_ores", joinListToString(aOres));
// oils
ArrayList tOilsTransformed = new ArrayList<>(aOils.size());
for (String aStr : aOils) {
String[] aStats = aStr.split(",");
tOilsTransformed.add(aStats[0] + ": " + aStats[1] + "L " + aStats[2]);
}
tNBT.setString("prospection_oils", joinListToString(tOilsTransformed));
String tOilsPosStr =
"X: " + Math.floorDiv(aX, 16 * 8) * 16 * 8 + " Z: " + Math.floorDiv(aZ, 16 * 8) * 16 * 8 + "\n";
int xOff = aX - Math.floorDiv(aX, 16 * 8) * 16 * 8;
xOff = xOff / 16;
int xOffRemain = 7 - xOff;
int zOff = aZ - Math.floorDiv(aZ, 16 * 8) * 16 * 8;
zOff = zOff / 16;
int zOffRemain = 7 - zOff;
for (; zOff > 0; zOff--) {
tOilsPosStr = tOilsPosStr.concat("--------\n");
}
for (; xOff > 0; xOff--) {
tOilsPosStr = tOilsPosStr.concat("-");
}
tOilsPosStr = tOilsPosStr.concat("P");
for (; xOffRemain > 0; xOffRemain--) {
tOilsPosStr = tOilsPosStr.concat("-");
}
tOilsPosStr = tOilsPosStr.concat("\n");
for (; zOffRemain > 0; zOffRemain--) {
tOilsPosStr = tOilsPosStr.concat("--------\n");
}
tOilsPosStr = tOilsPosStr.concat(" X: " + (Math.floorDiv(aX, 16 * 8) + 1) * 16 * 8 + " Z: "
+ (Math.floorDiv(aZ, 16 * 8) + 1) * 16 * 8); // +1 oilfied to find bottomright of [5]
tNBT.setString("prospection_oils_pos", tOilsPosStr);
tNBT.setString("prospection_radius", String.valueOf(aRadius));
setNBT(aStack, tNBT);
}
public static void convertProspectionData(ItemStack aStack) {
NBTTagCompound tNBT = getNBT(aStack);
byte tTier = tNBT.getByte("prospection_tier");
if (tTier == 0) { // basic prospection data
String tData = tNBT.getString("prospection");
String[] tDataArray = tData.split(",");
if (tDataArray.length > 6) {
tNBT.setString(
"author",
" Dim: " + tDataArray[3] + "X: " + tDataArray[0] + " Y: " + tDataArray[1] + " Z: "
+ tDataArray[2]);
NBTTagList tNBTList = new NBTTagList();
StringBuilder tOres = new StringBuilder(" Prospected Ores: ");
for (int i = 6; tDataArray.length > i; i++) {
tOres.append(tDataArray[i]).append(" ");
}
tNBTList.appendTag(new NBTTagString("Tier " + tTier + " Prospecting Data From: X" + tDataArray[0]
+ " Z:" + tDataArray[2] + " Dim:" + tDataArray[3] + " Produces " + tDataArray[4] + "L "
+ tDataArray[5] + " " + tOres));
tNBT.setTag("pages", tNBTList);
}
} else { // advanced prospection data
String tPos = tNBT.getString("prospection_pos");
String tRadius = tNBT.getString("prospection_radius");
String tOresStr = tNBT.getString("prospection_ores");
String tOilsStr = tNBT.getString("prospection_oils");
String tOilsPosStr = tNBT.getString("prospection_oils_pos");
String[] tOres = tOresStr.isEmpty() ? null : tOresStr.split("\\|");
String[] tOils = tOilsStr.isEmpty() ? null : tOilsStr.split("\\|");
NBTTagList tNBTList = new NBTTagList();
String tPageText = "Prospector report\n"
+ tPos + "\n\n"
+ "Oils: " + (tOils != null ? tOils.length : 0) + "\n\n"
+ "Ores within " + tRadius + " blocks\n\n"
+ "Location is center of orevein\n\n"
+ "Check NEI to confirm orevein type";
tNBTList.appendTag(new NBTTagString(tPageText));
if (tOres != null) fillBookWithList(tNBTList, "Ores Found %s\n\n", "\n", 7, tOres);
if (tOils != null) fillBookWithList(tNBTList, "Oils%s\n\n", "\n", 9, tOils);
tPageText = "Oil notes\n\n"
+ "Prospects from NW to SE 576 chunks"
+ "(9 8x8 oilfields)\n around and gives min-max amount" + "\n\n"
+ "[1][2][3]" + "\n"
+ "[4][5][6]" + "\n"
+ "[7][8][9]" + "\n"
+ "\n"
+ "[5] - Prospector in this 8x8 area";
tNBTList.appendTag(new NBTTagString(tPageText));
tPageText = "Corners of [5] are \n" + tOilsPosStr + "\n" + "P - Prospector in 8x8 field";
tNBTList.appendTag(new NBTTagString(tPageText));
tNBT.setString("author", tPos.replace("\n", " "));
tNBT.setTag("pages", tNBTList);
}
setNBT(aStack, tNBT);
}
public static void fillBookWithList(
NBTTagList aBook, String aPageHeader, String aListDelimiter, int aItemsPerPage, String[] list) {
String aPageFormatter = " %d/%d";
int tTotalPages = list.length / aItemsPerPage + (list.length % aItemsPerPage > 0 ? 1 : 0);
int tPage = 0;
StringBuilder tPageText;
do {
tPageText = new StringBuilder();
for (int i = tPage * aItemsPerPage; i < (tPage + 1) * aItemsPerPage && i < list.length; i += 1)
tPageText
.append((tPageText.length() == 0) ? "" : aListDelimiter)
.append(list[i]);
if (tPageText.length() > 0) {
String tPageCounter = tTotalPages > 1 ? String.format(aPageFormatter, tPage + 1, tTotalPages) : "";
NBTTagString tPageTag = new NBTTagString(String.format(aPageHeader, tPageCounter) + tPageText);
aBook.appendTag(tPageTag);
}
++tPage;
} while (tPageText.length() > 0);
}
public static void addEnchantment(ItemStack aStack, Enchantment aEnchantment, int aLevel) {
NBTTagCompound tNBT = getNBT(aStack), tEnchantmentTag;
if (!tNBT.hasKey("ench", 9)) tNBT.setTag("ench", new NBTTagList());
NBTTagList tList = tNBT.getTagList("ench", 10);
boolean temp = true;
for (int i = 0; i < tList.tagCount(); i++) {
tEnchantmentTag = tList.getCompoundTagAt(i);
if (tEnchantmentTag.getShort("id") == aEnchantment.effectId) {
tEnchantmentTag.setShort("id", (short) aEnchantment.effectId);
tEnchantmentTag.setShort("lvl", (byte) aLevel);
temp = false;
break;
}
}
if (temp) {
tEnchantmentTag = new NBTTagCompound();
tEnchantmentTag.setShort("id", (short) aEnchantment.effectId);
tEnchantmentTag.setShort("lvl", (byte) aLevel);
tList.appendTag(tEnchantmentTag);
}
aStack.setTagCompound(tNBT);
}
}
/**
* THIS IS BULLSHIT!!! WHY DO I HAVE TO DO THIS SHIT JUST TO HAVE ENCHANTS PROPERLY!?!
*/
public static class GT_EnchantmentHelper {
private static final BullshitIteratorA mBullshitIteratorA = new BullshitIteratorA();
private static final BullshitIteratorB mBullshitIteratorB = new BullshitIteratorB();
private static void applyBullshit(IBullshit aBullshitModifier, ItemStack aStack) {
if (aStack != null) {
NBTTagList nbttaglist = aStack.getEnchantmentTagList();
if (nbttaglist != null) {
try {
for (int i = 0; i < nbttaglist.tagCount(); ++i) {
short short1 = nbttaglist.getCompoundTagAt(i).getShort("id");
short short2 = nbttaglist.getCompoundTagAt(i).getShort("lvl");
if (Enchantment.enchantmentsList[short1] != null)
aBullshitModifier.calculateModifier(Enchantment.enchantmentsList[short1], short2);
}
} catch (Throwable e) {
/**/
}
}
}
}
private static void applyArrayOfBullshit(IBullshit aBullshitModifier, ItemStack[] aStacks) {
ItemStack[] aitemstack1 = aStacks;
int i = aStacks.length;
for (int j = 0; j < i; ++j) {
ItemStack itemstack = aitemstack1[j];
applyBullshit(aBullshitModifier, itemstack);
}
}
public static void applyBullshitA(EntityLivingBase aPlayer, Entity aEntity, ItemStack aStack) {
mBullshitIteratorA.mPlayer = aPlayer;
mBullshitIteratorA.mEntity = aEntity;
if (aPlayer != null) applyArrayOfBullshit(mBullshitIteratorA, aPlayer.getLastActiveItems());
if (aStack != null) applyBullshit(mBullshitIteratorA, aStack);
}
public static void applyBullshitB(EntityLivingBase aPlayer, Entity aEntity, ItemStack aStack) {
mBullshitIteratorB.mPlayer = aPlayer;
mBullshitIteratorB.mEntity = aEntity;
if (aPlayer != null) applyArrayOfBullshit(mBullshitIteratorB, aPlayer.getLastActiveItems());
if (aStack != null) applyBullshit(mBullshitIteratorB, aStack);
}
interface IBullshit {
void calculateModifier(Enchantment aEnchantment, int aLevel);
}
static final class BullshitIteratorA implements IBullshit {
public EntityLivingBase mPlayer;
public Entity mEntity;
BullshitIteratorA() {}
@Override
public void calculateModifier(Enchantment aEnchantment, int aLevel) {
aEnchantment.func_151367_b(mPlayer, mEntity, aLevel);
}
}
static final class BullshitIteratorB implements IBullshit {
public EntityLivingBase mPlayer;
public Entity mEntity;
BullshitIteratorB() {}
@Override
public void calculateModifier(Enchantment aEnchantment, int aLevel) {
aEnchantment.func_151368_a(mPlayer, mEntity, aLevel);
}
}
}
public static String toSubscript(long no) {
char[] chars = Long.toString(no).toCharArray();
for (int i = 0; i < chars.length; i++) {
chars[i] += 8272;
}
return new String(chars);
}
public static boolean isPartOfMaterials(ItemStack aStack, Materials aMaterials) {
return GT_OreDictUnificator.getAssociation(aStack) != null
&& GT_OreDictUnificator.getAssociation(aStack)
.mMaterial
.mMaterial
.equals(aMaterials);
}
public static boolean isPartOfOrePrefix(ItemStack aStack, OrePrefixes aPrefix) {
return GT_OreDictUnificator.getAssociation(aStack) != null
&& GT_OreDictUnificator.getAssociation(aStack).mPrefix.equals(aPrefix);
}
public static final ImmutableSet ORE_BLOCK_CLASSES = ImmutableSet.of(
"com.github.bartimaeusnek.bartworks.system.material.BW_MetaGenerated_Ores",
"com.github.bartimaeusnek.bartworks.system.material.BW_MetaGenerated_SmallOres",
"gtPlusPlus.core.block.base.BlockBaseOre");
public static boolean isOre(Block aBlock, int aMeta) {
return (aBlock instanceof GT_Block_Ores_Abstract)
|| isOre(new ItemStack(aBlock, 1, aMeta))
|| ORE_BLOCK_CLASSES.contains(aBlock.getClass().getName());
}
public static boolean isOre(ItemStack aStack) {
int tItem = GT_Utility.stackToInt(aStack);
if (sOreTable.containsKey(tItem)) {
return sOreTable.get(tItem);
}
for (int id : OreDictionary.getOreIDs(aStack)) {
if (OreDictionary.getOreName(id).startsWith("ore")) {
sOreTable.put(tItem, true);
return true;
}
}
sOreTable.put(tItem, false);
return false;
}
/**
* Do NOT mutate the returned {@code ItemStack}!
* We return {@code ItemStack} instead of {@code Block} so that we can include metadata.
*/
public static ItemStack getCobbleForOre(Block ore, short metaData) {
// We need to convert small ores to regular ores because small ores don't have associated ItemData.
// We take the modulus of the metadata by 16000 because that is the magic number to convert small ores to
// regular ores.
// See: GT_TileEntity_Ores.java
ItemData association =
GT_OreDictUnificator.getAssociation(new ItemStack(Item.getItemFromBlock(ore), 1, metaData % 16000));
if (association != null) {
Supplier supplier = sOreToCobble.get(association.mPrefix);
if (supplier != null) {
return supplier.get();
}
}
return new ItemStack(Blocks.cobblestone);
}
public static Optional reverseShapelessRecipe(ItemStack output, Object... aRecipe) {
if (output == null) {
return Optional.empty();
}
List inputs = new ArrayList<>();
for (Object o : aRecipe) {
if (o instanceof ItemStack) {
ItemStack toAdd = ((ItemStack) o).copy();
inputs.add(toAdd);
} else if (o instanceof String) {
ItemStack stack = GT_OreDictUnificator.get(o, 1);
if (stack == null) {
Optional oStack =
OreDictionary.getOres((String) o).stream().findAny();
if (oStack.isPresent()) {
ItemStack copy = oStack.get().copy();
inputs.add(copy);
}
} else {
ItemStack copy = stack.copy();
inputs.add(copy);
}
} else if (o instanceof Item) inputs.add(new ItemStack((Item) o));
else if (o instanceof Block) inputs.add(new ItemStack((Block) o));
else throw new IllegalStateException("A Recipe contains an invalid input! Output: " + output);
}
inputs.removeIf(x -> x.getItem() instanceof GT_MetaGenerated_Tool);
return Optional.of(new GT_Recipe(
false, new ItemStack[] {output}, inputs.toArray(new ItemStack[0]), null, null, null, null, 300, 30, 0));
}
public static Optional reverseShapedRecipe(ItemStack output, Object... aRecipe) {
if (output == null) {
return Optional.empty();
}
Map