mirror of
				https://github.com/SquidDev-CC/CC-Tweaked
				synced 2025-10-26 11:27:38 +00:00 
			
		
		
		
	Make upgrade recipe requirements a little more lax
- Move some common upgrade code to IUpgradeBase. 99% sure this this preserves binary compatibility (on the JVM at least). - Instead of requiring the share tag to match, allow upgrades to specify their own predicate. IMO this is a little ugly, but required to fix #614 as other mods chuck their own NBT on items.
This commit is contained in:
		
							
								
								
									
										81
									
								
								src/main/java/dan200/computercraft/api/IUpgradeBase.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										81
									
								
								src/main/java/dan200/computercraft/api/IUpgradeBase.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,81 @@ | |||||||
|  | package dan200.computercraft.api; | ||||||
|  |  | ||||||
|  | import dan200.computercraft.api.pocket.IPocketUpgrade; | ||||||
|  | import dan200.computercraft.api.turtle.ITurtleUpgrade; | ||||||
|  | import net.minecraft.item.ItemStack; | ||||||
|  | import net.minecraft.nbt.CompoundNBT; | ||||||
|  | import net.minecraft.util.ResourceLocation; | ||||||
|  |  | ||||||
|  | import javax.annotation.Nonnull; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Common functionality between {@link ITurtleUpgrade} and {@link IPocketUpgrade}. | ||||||
|  |  */ | ||||||
|  | public interface IUpgradeBase | ||||||
|  | { | ||||||
|  |     /** | ||||||
|  |      * Gets a unique identifier representing this type of turtle upgrade. eg: "computercraft:wireless_modem" | ||||||
|  |      * or "my_mod:my_upgrade". | ||||||
|  |      * | ||||||
|  |      * You should use a unique resource domain to ensure this upgrade is uniquely identified. | ||||||
|  |      * The upgrade will fail registration if an already used ID is specified. | ||||||
|  |      * | ||||||
|  |      * @return The unique ID for this upgrade. | ||||||
|  |      */ | ||||||
|  |     @Nonnull | ||||||
|  |     ResourceLocation getUpgradeID(); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Return an unlocalised string to describe this type of computer in item names. | ||||||
|  |      * | ||||||
|  |      * Examples of built-in adjectives are "Wireless", "Mining" and "Crafty". | ||||||
|  |      * | ||||||
|  |      * @return The localisation key for this upgrade's adjective. | ||||||
|  |      */ | ||||||
|  |     @Nonnull | ||||||
|  |     String getUnlocalisedAdjective(); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Return an item stack representing the type of item that a computer must be crafted | ||||||
|  |      * with to create a version which holds this upgrade. This item stack is also used | ||||||
|  |      * to determine the upgrade given by {@code turtle.equipLeft()} or {@code pocket.equipBack()} | ||||||
|  |      * | ||||||
|  |      * This should be constant over a session (or at least a datapack reload). It is recommended | ||||||
|  |      * that you cache the stack too, in order to prevent constructing it every time the method | ||||||
|  |      * is called. | ||||||
|  |      * | ||||||
|  |      * @return The item stack to craft with, or {@link ItemStack#EMPTY} if it cannot be crafted. | ||||||
|  |      */ | ||||||
|  |     @Nonnull | ||||||
|  |     ItemStack getCraftingItem(); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Determine if an item is suitable for being used for this upgrade. | ||||||
|  |      * | ||||||
|  |      * When un-equipping an upgrade, we return {@link #getCraftingItem()} rather than | ||||||
|  |      * the original stack. In order to prevent people losing items with enchantments (or | ||||||
|  |      * repairing items with non-0 damage), we impose additional checks on the item. | ||||||
|  |      * | ||||||
|  |      * The default check requires that any non-capability NBT is exactly the same as the | ||||||
|  |      * crafting item, but this may be relaxed for your upgrade. | ||||||
|  |      * | ||||||
|  |      * @param stack The stack to check. This is guaranteed to be non-empty and have the same item as | ||||||
|  |      *              {@link #getCraftingItem()}. | ||||||
|  |      * @return If this stack may be used to equip this upgrade. | ||||||
|  |      * @see net.minecraftforge.common.crafting.NBTIngredient#test(ItemStack) For the implementation of the default | ||||||
|  |      * check. | ||||||
|  |      */ | ||||||
|  |     default boolean isItemSuitable( @Nonnull ItemStack stack ) | ||||||
|  |     { | ||||||
|  |         ItemStack crafting = getCraftingItem(); | ||||||
|  |  | ||||||
|  |         // A more expanded form of ItemStack.areShareTagsEqual, but allowing an empty tag to be equal to a | ||||||
|  |         // null one. | ||||||
|  |         CompoundNBT shareTag = stack.getItem().getShareTag( stack ); | ||||||
|  |         CompoundNBT craftingShareTag = crafting.getItem().getShareTag( crafting ); | ||||||
|  |         if( shareTag == craftingShareTag ) return true; | ||||||
|  |         if( shareTag == null ) return craftingShareTag.isEmpty(); | ||||||
|  |         if( craftingShareTag == null ) return shareTag.isEmpty(); | ||||||
|  |         return shareTag.equals( craftingShareTag ); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -6,10 +6,8 @@ | |||||||
| package dan200.computercraft.api.pocket; | package dan200.computercraft.api.pocket; | ||||||
|  |  | ||||||
| import dan200.computercraft.api.ComputerCraftAPI; | import dan200.computercraft.api.ComputerCraftAPI; | ||||||
|  | import dan200.computercraft.api.IUpgradeBase; | ||||||
| import dan200.computercraft.api.peripheral.IPeripheral; | import dan200.computercraft.api.peripheral.IPeripheral; | ||||||
| import dan200.computercraft.api.turtle.ITurtleUpgrade; |  | ||||||
| import net.minecraft.item.ItemStack; |  | ||||||
| import net.minecraft.util.ResourceLocation; |  | ||||||
| import net.minecraft.world.World; | import net.minecraft.world.World; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| @@ -18,49 +16,10 @@ import javax.annotation.Nullable; | |||||||
| /** | /** | ||||||
|  * Additional peripherals for pocket computers. |  * Additional peripherals for pocket computers. | ||||||
|  * |  * | ||||||
|  * This is similar to {@link ITurtleUpgrade}. |  * @see ComputerCraftAPI#registerPocketUpgrade(IPocketUpgrade) | ||||||
|  */ |  */ | ||||||
| public interface IPocketUpgrade | public interface IPocketUpgrade extends IUpgradeBase | ||||||
| { | { | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Gets a unique identifier representing this type of turtle upgrade. eg: "computercraft:wireless_modem" or |  | ||||||
|      * "my_mod:my_upgrade". |  | ||||||
|      * |  | ||||||
|      * You should use a unique resource domain to ensure this upgrade is uniquely identified. The upgrade will fail |  | ||||||
|      * registration if an already used ID is specified. |  | ||||||
|      * |  | ||||||
|      * @return The upgrade's id. |  | ||||||
|      * @see IPocketUpgrade#getUpgradeID() |  | ||||||
|      * @see ComputerCraftAPI#registerPocketUpgrade(IPocketUpgrade) |  | ||||||
|      */ |  | ||||||
|     @Nonnull |  | ||||||
|     ResourceLocation getUpgradeID(); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Return an unlocalised string to describe the type of pocket computer this upgrade provides. |  | ||||||
|      * |  | ||||||
|      * An example of a built-in adjectives is "Wireless" - this is converted to "Wireless Pocket Computer". |  | ||||||
|      * |  | ||||||
|      * @return The unlocalised adjective. |  | ||||||
|      * @see ITurtleUpgrade#getUnlocalisedAdjective() |  | ||||||
|      */ |  | ||||||
|     @Nonnull |  | ||||||
|     String getUnlocalisedAdjective(); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Return an item stack representing the type of item that a pocket computer must be crafted with to create a |  | ||||||
|      * pocket computer which holds this upgrade. This item stack is also used to determine the upgrade given by |  | ||||||
|      * {@code pocket.equip()}/{@code pocket.unequip()}. |  | ||||||
|      * |  | ||||||
|      * Ideally this should be constant over a session. It is recommended that you cache |  | ||||||
|      * the item too, in order to prevent constructing it every time the method is called. |  | ||||||
|      * |  | ||||||
|      * @return The item stack used for crafting. This can be {@link ItemStack#EMPTY} if crafting is disabled. |  | ||||||
|      */ |  | ||||||
|     @Nonnull |  | ||||||
|     ItemStack getCraftingItem(); |  | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Creates a peripheral for the pocket computer. |      * Creates a peripheral for the pocket computer. | ||||||
|      * |      * | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ | |||||||
| package dan200.computercraft.api.turtle; | package dan200.computercraft.api.turtle; | ||||||
|  |  | ||||||
| import dan200.computercraft.api.ComputerCraftAPI; | import dan200.computercraft.api.ComputerCraftAPI; | ||||||
|  | import dan200.computercraft.api.IUpgradeBase; | ||||||
| import dan200.computercraft.api.client.TransformedModel; | import dan200.computercraft.api.client.TransformedModel; | ||||||
| import dan200.computercraft.api.peripheral.IPeripheral; | import dan200.computercraft.api.peripheral.IPeripheral; | ||||||
| import dan200.computercraft.api.turtle.event.TurtleAttackEvent; | import dan200.computercraft.api.turtle.event.TurtleAttackEvent; | ||||||
| @@ -13,7 +14,6 @@ import dan200.computercraft.api.turtle.event.TurtleBlockEvent; | |||||||
| import net.minecraft.client.renderer.model.ModelResourceLocation; | import net.minecraft.client.renderer.model.ModelResourceLocation; | ||||||
| import net.minecraft.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
| import net.minecraft.util.Direction; | import net.minecraft.util.Direction; | ||||||
| import net.minecraft.util.ResourceLocation; |  | ||||||
| import net.minecraftforge.api.distmarker.Dist; | import net.minecraftforge.api.distmarker.Dist; | ||||||
| import net.minecraftforge.api.distmarker.OnlyIn; | import net.minecraftforge.api.distmarker.OnlyIn; | ||||||
| import net.minecraftforge.event.entity.player.AttackEntityEvent; | import net.minecraftforge.event.entity.player.AttackEntityEvent; | ||||||
| @@ -28,29 +28,8 @@ import javax.annotation.Nullable; | |||||||
|  * |  * | ||||||
|  * @see ComputerCraftAPI#registerTurtleUpgrade(ITurtleUpgrade) |  * @see ComputerCraftAPI#registerTurtleUpgrade(ITurtleUpgrade) | ||||||
|  */ |  */ | ||||||
| public interface ITurtleUpgrade | public interface ITurtleUpgrade extends IUpgradeBase | ||||||
| { | { | ||||||
|     /** |  | ||||||
|      * Gets a unique identifier representing this type of turtle upgrade. eg: "computercraft:wireless_modem" or "my_mod:my_upgrade". |  | ||||||
|      * You should use a unique resource domain to ensure this upgrade is uniquely identified. |  | ||||||
|      * The turtle will fail registration if an already used ID is specified. |  | ||||||
|      * |  | ||||||
|      * @return The unique ID for this upgrade. |  | ||||||
|      * @see ComputerCraftAPI#registerTurtleUpgrade(ITurtleUpgrade) |  | ||||||
|      */ |  | ||||||
|     @Nonnull |  | ||||||
|     ResourceLocation getUpgradeID(); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Return an unlocalised string to describe this type of turtle in turtle item names. |  | ||||||
|      * |  | ||||||
|      * Examples of built-in adjectives are "Wireless", "Mining" and "Crafty". |  | ||||||
|      * |  | ||||||
|      * @return The localisation key for this upgrade's adjective. |  | ||||||
|      */ |  | ||||||
|     @Nonnull |  | ||||||
|     String getUnlocalisedAdjective(); |  | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Return whether this turtle adds a tool or a peripheral to the turtle. |      * Return whether this turtle adds a tool or a peripheral to the turtle. | ||||||
|      * |      * | ||||||
| @@ -60,19 +39,6 @@ public interface ITurtleUpgrade | |||||||
|     @Nonnull |     @Nonnull | ||||||
|     TurtleUpgradeType getType(); |     TurtleUpgradeType getType(); | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Return an item stack representing the type of item that a turtle must be crafted |  | ||||||
|      * with to create a turtle which holds this upgrade. This item stack is also used |  | ||||||
|      * to determine the upgrade given by {@code turtle.equip()} |  | ||||||
|      * |  | ||||||
|      * Ideally this should be constant over a session. It is recommended that you cache |  | ||||||
|      * the item too, in order to prevent constructing it every time the method is called. |  | ||||||
|      * |  | ||||||
|      * @return The item stack to craft with, or {@link ItemStack#EMPTY} if it cannot be crafted. |  | ||||||
|      */ |  | ||||||
|     @Nonnull |  | ||||||
|     ItemStack getCraftingItem(); |  | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Will only be called for peripheral upgrades. Creates a peripheral for a turtle being placed using this upgrade. |      * Will only be called for peripheral upgrades. Creates a peripheral for a turtle being placed using this upgrade. | ||||||
|      * |      * | ||||||
|   | |||||||
| @@ -7,7 +7,6 @@ package dan200.computercraft.shared; | |||||||
|  |  | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
| import dan200.computercraft.api.pocket.IPocketUpgrade; | import dan200.computercraft.api.pocket.IPocketUpgrade; | ||||||
| import dan200.computercraft.shared.util.InventoryUtil; |  | ||||||
| import net.minecraft.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
| import net.minecraftforge.fml.ModContainer; | import net.minecraftforge.fml.ModContainer; | ||||||
| import net.minecraftforge.fml.ModLoadingContext; | import net.minecraftforge.fml.ModLoadingContext; | ||||||
| @@ -55,7 +54,7 @@ public final class PocketUpgrades | |||||||
|         for( IPocketUpgrade upgrade : upgrades.values() ) |         for( IPocketUpgrade upgrade : upgrades.values() ) | ||||||
|         { |         { | ||||||
|             ItemStack craftingStack = upgrade.getCraftingItem(); |             ItemStack craftingStack = upgrade.getCraftingItem(); | ||||||
|             if( !craftingStack.isEmpty() && InventoryUtil.areItemsSimilar( stack, craftingStack ) ) |             if( !craftingStack.isEmpty() && craftingStack.getItem() == stack.getItem() && upgrade.isItemSuitable( stack ) ) | ||||||
|             { |             { | ||||||
|                 return upgrade; |                 return upgrade; | ||||||
|             } |             } | ||||||
|   | |||||||
| @@ -8,7 +8,6 @@ package dan200.computercraft.shared; | |||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
| import dan200.computercraft.api.turtle.ITurtleUpgrade; | import dan200.computercraft.api.turtle.ITurtleUpgrade; | ||||||
| import dan200.computercraft.shared.computer.core.ComputerFamily; | import dan200.computercraft.shared.computer.core.ComputerFamily; | ||||||
| import dan200.computercraft.shared.util.InventoryUtil; |  | ||||||
| import net.minecraft.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
| import net.minecraftforge.fml.ModLoadingContext; | import net.minecraftforge.fml.ModLoadingContext; | ||||||
|  |  | ||||||
| @@ -83,7 +82,7 @@ public final class TurtleUpgrades | |||||||
|             if( !wrapper.enabled ) continue; |             if( !wrapper.enabled ) continue; | ||||||
|  |  | ||||||
|             ItemStack craftingStack = wrapper.upgrade.getCraftingItem(); |             ItemStack craftingStack = wrapper.upgrade.getCraftingItem(); | ||||||
|             if( !craftingStack.isEmpty() && InventoryUtil.areItemsSimilar( stack, craftingStack ) ) |             if( !craftingStack.isEmpty() && craftingStack.getItem() == stack.getItem() && wrapper.upgrade.isItemSuitable( stack ) ) | ||||||
|             { |             { | ||||||
|                 return wrapper.upgrade; |                 return wrapper.upgrade; | ||||||
|             } |             } | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ | |||||||
| package dan200.computercraft.shared.integration.jei; | package dan200.computercraft.shared.integration.jei; | ||||||
|  |  | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
|  | import dan200.computercraft.api.IUpgradeBase; | ||||||
| import dan200.computercraft.api.pocket.IPocketUpgrade; | import dan200.computercraft.api.pocket.IPocketUpgrade; | ||||||
| import dan200.computercraft.api.turtle.ITurtleUpgrade; | import dan200.computercraft.api.turtle.ITurtleUpgrade; | ||||||
| import dan200.computercraft.api.turtle.TurtleSide; | import dan200.computercraft.api.turtle.TurtleSide; | ||||||
| @@ -16,7 +17,6 @@ import dan200.computercraft.shared.pocket.items.ItemPocketComputer; | |||||||
| import dan200.computercraft.shared.pocket.items.PocketComputerItemFactory; | import dan200.computercraft.shared.pocket.items.PocketComputerItemFactory; | ||||||
| import dan200.computercraft.shared.turtle.items.ITurtleItem; | import dan200.computercraft.shared.turtle.items.ITurtleItem; | ||||||
| import dan200.computercraft.shared.turtle.items.TurtleItemFactory; | import dan200.computercraft.shared.turtle.items.TurtleItemFactory; | ||||||
| import dan200.computercraft.shared.util.InventoryUtil; |  | ||||||
| import mezz.jei.api.constants.VanillaRecipeCategoryUid; | import mezz.jei.api.constants.VanillaRecipeCategoryUid; | ||||||
| import mezz.jei.api.recipe.IFocus; | import mezz.jei.api.recipe.IFocus; | ||||||
| import mezz.jei.api.recipe.advanced.IRecipeManagerPlugin; | import mezz.jei.api.recipe.advanced.IRecipeManagerPlugin; | ||||||
| @@ -83,7 +83,10 @@ class RecipeResolver implements IRecipeManagerPlugin | |||||||
|         for( UpgradeInfo upgrade : upgrades ) |         for( UpgradeInfo upgrade : upgrades ) | ||||||
|         { |         { | ||||||
|             ItemStack craftingStack = upgrade.stack; |             ItemStack craftingStack = upgrade.stack; | ||||||
|             if( !craftingStack.isEmpty() && InventoryUtil.areItemsSimilar( stack, craftingStack ) ) return true; |             if( !craftingStack.isEmpty() && craftingStack.getItem() == stack.getItem() && upgrade.upgrade.isItemSuitable( stack ) ) | ||||||
|  |             { | ||||||
|  |                 return true; | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         return false; |         return false; | ||||||
| @@ -194,11 +197,10 @@ class RecipeResolver implements IRecipeManagerPlugin | |||||||
|  |  | ||||||
|             List<Shaped> recipes = null; |             List<Shaped> recipes = null; | ||||||
|             boolean multiple = false; |             boolean multiple = false; | ||||||
|             Ingredient ingredient = fromStacks( stack ); |  | ||||||
|             for( UpgradeInfo upgrade : upgrades ) |             for( UpgradeInfo upgrade : upgrades ) | ||||||
|             { |             { | ||||||
|                 ItemStack craftingStack = upgrade.stack; |                 ItemStack craftingStack = upgrade.stack; | ||||||
|                 if( craftingStack.isEmpty() || !InventoryUtil.areItemsSimilar( stack, craftingStack ) ) |                 if( !craftingStack.isEmpty() && craftingStack.getItem() == stack.getItem() && upgrade.upgrade.isItemSuitable( stack ) ) | ||||||
|                 { |                 { | ||||||
|                     continue; |                     continue; | ||||||
|                 } |                 } | ||||||
| @@ -332,42 +334,29 @@ class RecipeResolver implements IRecipeManagerPlugin | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static final class Upgrade<T> |  | ||||||
|     { |  | ||||||
|         final T upgrade; |  | ||||||
|         final ItemStack stack; |  | ||||||
|         final Ingredient ingredient; |  | ||||||
|  |  | ||||||
|         private Upgrade( T upgrade, ItemStack stack ) |  | ||||||
|         { |  | ||||||
|             this.upgrade = upgrade; |  | ||||||
|             this.stack = stack; |  | ||||||
|             ingredient = fromStacks( stack ); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private static class UpgradeInfo |     private static class UpgradeInfo | ||||||
|     { |     { | ||||||
|         final ItemStack stack; |         final ItemStack stack; | ||||||
|         final Ingredient ingredient; |         final Ingredient ingredient; | ||||||
|         final ITurtleUpgrade turtle; |         final ITurtleUpgrade turtle; | ||||||
|         final IPocketUpgrade pocket; |         final IPocketUpgrade pocket; | ||||||
|  |         final IUpgradeBase upgrade; | ||||||
|         ArrayList<Shaped> recipes; |         ArrayList<Shaped> recipes; | ||||||
|  |  | ||||||
|         UpgradeInfo( ItemStack stack, ITurtleUpgrade turtle ) |         UpgradeInfo( ItemStack stack, ITurtleUpgrade turtle ) | ||||||
|         { |         { | ||||||
|             this.stack = stack; |             this.stack = stack; | ||||||
|             ingredient = fromStacks( stack ); |             this.ingredient = fromStacks( stack ); | ||||||
|             this.turtle = turtle; |             this.upgrade = this.turtle = turtle; | ||||||
|             pocket = null; |             this.pocket = null; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         UpgradeInfo( ItemStack stack, IPocketUpgrade pocket ) |         UpgradeInfo( ItemStack stack, IPocketUpgrade pocket ) | ||||||
|         { |         { | ||||||
|             this.stack = stack; |             this.stack = stack; | ||||||
|             ingredient = fromStacks( stack ); |             this.ingredient = fromStacks( stack ); | ||||||
|             turtle = null; |             this.turtle = null; | ||||||
|             this.pocket = pocket; |             this.upgrade = this.pocket = pocket; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         List<Shaped> getRecipes() |         List<Shaped> getRecipes() | ||||||
|   | |||||||
| @@ -28,6 +28,7 @@ import net.minecraft.entity.item.ArmorStandEntity; | |||||||
| import net.minecraft.fluid.IFluidState; | import net.minecraft.fluid.IFluidState; | ||||||
| import net.minecraft.item.Item; | import net.minecraft.item.Item; | ||||||
| import net.minecraft.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
|  | import net.minecraft.nbt.CompoundNBT; | ||||||
| import net.minecraft.tileentity.TileEntity; | import net.minecraft.tileentity.TileEntity; | ||||||
| import net.minecraft.util.DamageSource; | import net.minecraft.util.DamageSource; | ||||||
| import net.minecraft.util.Direction; | import net.minecraft.util.Direction; | ||||||
| @@ -38,6 +39,7 @@ import net.minecraft.world.World; | |||||||
| import net.minecraftforge.api.distmarker.Dist; | import net.minecraftforge.api.distmarker.Dist; | ||||||
| import net.minecraftforge.api.distmarker.OnlyIn; | import net.minecraftforge.api.distmarker.OnlyIn; | ||||||
| import net.minecraftforge.common.MinecraftForge; | import net.minecraftforge.common.MinecraftForge; | ||||||
|  | import net.minecraftforge.common.util.Constants; | ||||||
| import net.minecraftforge.event.entity.player.AttackEntityEvent; | import net.minecraftforge.event.entity.player.AttackEntityEvent; | ||||||
| import net.minecraftforge.event.world.BlockEvent; | import net.minecraftforge.event.world.BlockEvent; | ||||||
| import org.apache.commons.lang3.tuple.Pair; | import org.apache.commons.lang3.tuple.Pair; | ||||||
| @@ -68,6 +70,24 @@ public class TurtleTool extends AbstractTurtleUpgrade | |||||||
|         this.item = toolItem; |         this.item = toolItem; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean isItemSuitable( @Nonnull ItemStack stack ) | ||||||
|  |     { | ||||||
|  |         CompoundNBT tag = stack.getTag(); | ||||||
|  |         if( tag == null || tag.isEmpty() ) return true; | ||||||
|  |  | ||||||
|  |         // Check we've not got anything vaguely interesting on the item. We allow other mods to add their | ||||||
|  |         // own NBT, with the understanding such details will be lost to the mist of time. | ||||||
|  |         if( stack.isDamaged() || stack.isEnchanted() || stack.hasDisplayName() ) return false; | ||||||
|  |         if( tag.contains( "AttributeModifiers", Constants.NBT.TAG_LIST ) && | ||||||
|  |             !tag.getList( "AttributeModifiers", Constants.NBT.TAG_COMPOUND ).isEmpty() ) | ||||||
|  |         { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     @Override |     @Override | ||||||
|     @OnlyIn( Dist.CLIENT ) |     @OnlyIn( Dist.CLIENT ) | ||||||
|   | |||||||
| @@ -9,7 +9,6 @@ import net.minecraft.entity.Entity; | |||||||
| import net.minecraft.inventory.IInventory; | import net.minecraft.inventory.IInventory; | ||||||
| import net.minecraft.inventory.ISidedInventory; | import net.minecraft.inventory.ISidedInventory; | ||||||
| import net.minecraft.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
| import net.minecraft.nbt.CompoundNBT; |  | ||||||
| import net.minecraft.tileentity.TileEntity; | import net.minecraft.tileentity.TileEntity; | ||||||
| import net.minecraft.util.Direction; | import net.minecraft.util.Direction; | ||||||
| import net.minecraft.util.math.BlockPos; | import net.minecraft.util.math.BlockPos; | ||||||
| @@ -40,35 +39,6 @@ public final class InventoryUtil | |||||||
|         return a == b || ItemHandlerHelper.canItemStacksStack( a, b ); |         return a == b || ItemHandlerHelper.canItemStacksStack( a, b ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Determines if two items are "mostly" equivalent. Namely, they have the same item and damage, and identical |  | ||||||
|      * share stacks. |  | ||||||
|      * |  | ||||||
|      * This is largely based on {@link net.minecraftforge.common.crafting.IngredientNBT#test(ItemStack)}. It is |  | ||||||
|      * sufficient to ensure basic information (such as enchantments) are the same, while not having to worry about |  | ||||||
|      * capabilities. |  | ||||||
|      * |  | ||||||
|      * @param a The first stack to check |  | ||||||
|      * @param b The second stack to check |  | ||||||
|      * @return If these items are largely the same. |  | ||||||
|      */ |  | ||||||
|     public static boolean areItemsSimilar( @Nonnull ItemStack a, @Nonnull ItemStack b ) |  | ||||||
|     { |  | ||||||
|         if( a == b ) return true; |  | ||||||
|         if( a.isEmpty() ) return !b.isEmpty(); |  | ||||||
|  |  | ||||||
|         if( a.getItem() != b.getItem() ) return false; |  | ||||||
|  |  | ||||||
|         // A more expanded form of ItemStack.areShareTagsEqual, but allowing an empty tag to be equal to a |  | ||||||
|         // null one. |  | ||||||
|         CompoundNBT shareTagA = a.getItem().getShareTag( a ); |  | ||||||
|         CompoundNBT shareTagB = b.getItem().getShareTag( b ); |  | ||||||
|         if( shareTagA == shareTagB ) return true; |  | ||||||
|         if( shareTagA == null ) return shareTagB.isEmpty(); |  | ||||||
|         if( shareTagB == null ) return shareTagA.isEmpty(); |  | ||||||
|         return shareTagA.equals( shareTagB ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // Methods for finding inventories: |     // Methods for finding inventories: | ||||||
|  |  | ||||||
|     public static IItemHandler getInventory( World world, BlockPos pos, Direction side ) |     public static IItemHandler getInventory( World world, BlockPos pos, Direction side ) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Jonathan Coates
					Jonathan Coates