diff --git a/README.MD b/README.MD index 3e501b9..0f43b1a 100644 --- a/README.MD +++ b/README.MD @@ -9,7 +9,8 @@ You do more damage to mobs (25%). **Your superfoods are glow berries and apples.** 4. Dog You get speed 2 when doing damage. **Your superfood are uncooked meats.** - +5. Cow + When you hit a mob, there's a 30% chance of slowing them down. Works on players aswell. **Your superfoods are apples, golden apples, and carrots.** ## **Chat** Chat while you are a animal is very different. When you speak, your words will become furry-ified and every time you talk you'll have animal sounds come out of you. You can disable this via /chatmodoff, and turn it back on via /chatmodon. diff --git a/src/main/java/ovh/sad/animalrp/AnimalRP.java b/src/main/java/ovh/sad/animalrp/AnimalRP.java index cbac811..e6fe243 100644 --- a/src/main/java/ovh/sad/animalrp/AnimalRP.java +++ b/src/main/java/ovh/sad/animalrp/AnimalRP.java @@ -1,6 +1,9 @@ package ovh.sad.animalrp; import eu.pb4.placeholders.api.*; +import eu.pb4.placeholders.api.parsers.NodeParser; +import eu.pb4.placeholders.api.parsers.TagParser; +import eu.pb4.placeholders.api.parsers.TextParserV1; import net.fabricmc.api.ModInitializer; import net.fabricmc.fabric.api.command.v2.CommandRegistrationCallback; import net.fabricmc.fabric.api.entity.event.v1.ServerLivingEntityEvents; @@ -12,13 +15,10 @@ import net.fabricmc.fabric.api.event.player.UseEntityCallback; import net.minecraft.entity.player.PlayerEntity; import net.minecraft.server.network.ServerPlayerEntity; import net.minecraft.sound.SoundCategory; +import net.minecraft.text.Text; import net.minecraft.util.ActionResult; import net.minecraft.util.Identifier; -import ovh.sad.animalrp.animals.Animal; -import ovh.sad.animalrp.animals.Cat; -import ovh.sad.animalrp.animals.Dog; -import ovh.sad.animalrp.animals.Fox; -import ovh.sad.animalrp.animals.Bee; +import ovh.sad.animalrp.animals.*; import ovh.sad.animalrp.commands.InfoCommand; import ovh.sad.animalrp.commands.InteractionCommand; import ovh.sad.animalrp.commands.NoChatCommand; @@ -49,7 +49,11 @@ public class AnimalRP implements ModInitializer { private static final Map lastPetTimes = new ConcurrentHashMap<>(); private static final long COOLDOWN_MS = 1000; + public static final NodeParser parser = TagParser.DEFAULT; + public static Text parseText(String text) { + return AnimalRP.parser.parseNode(text).toText(); + } @Override public void onInitialize() { @@ -61,11 +65,12 @@ ZZZzz /,`.-'`' -. ;-;;,_ """; LOGGER.info(catAscii); LOGGER.info("furry animal mod for fabric"); - LOGGER.info("by @fucksophie - est. Sept 2024"); + LOGGER.info("by @fucksophie - est. Sept 2024, improvements Mar 2025"); animals.put("cat", new Cat()); animals.put("dog", new Dog()); animals.put("fox", new Fox()); animals.put("bee", new Bee()); + animals.put("cow", new Cow()); HashmapStore.get("users.json").forEach((k, v) -> { users.put(UUID.fromString(k), animals.get(v)); @@ -131,14 +136,14 @@ ZZZzz /,`.-'`' -. ;-;;,_ if (lastPetTimes.getOrDefault(attackerId, 0L) + COOLDOWN_MS > currentTime) { return ActionResult.PASS; } + lastPetTimes.put(attackerId, currentTime); - - attackee.sendMessage(TextParserUtils.formatText( + attackee.sendMessage(AnimalRP.parseText( String.format("%s petted you! %s", "" + attacker.getName().getString() + "", "" + attackerAnimal.catchphrase)), false); - attacker.sendMessage(TextParserUtils.formatText( + attacker.sendMessage(AnimalRP.parseText( String.format("You petted %s! %s", "" + attackee.getName().getString() + "", "" + attackeeAnimal.catchphrase)), false); diff --git a/src/main/java/ovh/sad/animalrp/animals/Bee.java b/src/main/java/ovh/sad/animalrp/animals/Bee.java index 64c81b3..4c52127 100644 --- a/src/main/java/ovh/sad/animalrp/animals/Bee.java +++ b/src/main/java/ovh/sad/animalrp/animals/Bee.java @@ -26,6 +26,7 @@ import net.minecraft.sound.SoundCategory; import net.minecraft.sound.SoundEvents; import net.minecraft.util.ActionResult; import net.minecraft.util.Identifier; +import org.jetbrains.annotations.NotNull; import ovh.sad.animalrp.AnimalRP; import ovh.sad.animalrp.util.Mood; import ovh.sad.animalrp.util.TextDestroyer; @@ -37,13 +38,13 @@ public class Bee extends Animal { } - private static Item[] _allFlowers = { Items.ALLIUM, Items.AZURE_BLUET, Items.BLUE_ORCHID, + private static final Item[] _allFlowers = { Items.ALLIUM, Items.AZURE_BLUET, Items.BLUE_ORCHID, Items.CORNFLOWER, Items.DANDELION, Items.LILY_OF_THE_VALLEY, Items.OXEYE_DAISY, Items.POPPY, Items.TORCHFLOWER, Items.ORANGE_TULIP, Items.PINK_TULIP, Items.RED_TULIP, Items.WHITE_TULIP }; - public static List allFlowers = Arrays.asList(_allFlowers); - public static Identifier beeFoodKey = Identifier.of("animalrp", "bee_food"); - public static HashMap inARow = new HashMap(); + public static final List allFlowers = Arrays.asList(_allFlowers); + public static HashMap inARow = new HashMap<>(); + TextDestroyer destroyer = new TextDestroyer(new String[] { ">_<", "*buzz*", ";3", ":3", "εწз", " ≧◠◡◠≦ ", "*stings you*", "*humms*", @@ -63,7 +64,7 @@ public class Bee extends Animal { ArrayList sneakers = new ArrayList(); - public static Boolean isItemARP(ItemStack is) { + public static @NotNull Boolean isItemARP(ItemStack is) { return (allFlowers.contains(is.getItem()) && is.get(DataComponentTypes.FOOD) != null); } @@ -89,12 +90,12 @@ public class Bee extends Animal { return ActionResult.PASS; } - Boolean incorrect = false; + boolean incorrect = false; if (animal == null) { incorrect = true; } else { - if (animal.name != this.name) { + if (!animal.name.equals(this.name)) { incorrect = true; } } @@ -129,7 +130,7 @@ public class Bee extends Animal { // Called from the FoodEating mixin. // Called only if the player is a animal and a bee. - public void onEat(ServerPlayerEntity player, ItemStack item) { + public void onEat(ServerPlayerEntity player, @NotNull ItemStack item) { if (!allFlowers.contains(item.getItem())) { // not a flower return; } @@ -163,11 +164,11 @@ public class Bee extends Animal { } // Called from the 'Sneaking' mixin. - public void onSneak(ServerPlayerEntity player, boolean status) { + public void onSneak(@NotNull ServerPlayerEntity player, boolean status) { Animal animal = AnimalRP.users.get(player.getUuid()); if (animal == null) return; - if (animal.name != this.name) + if (!animal.name.equals(this.name)) return; Block type = player.getWorld().getBlockState(player.getBlockPos().down()).getBlock(); diff --git a/src/main/java/ovh/sad/animalrp/animals/Cow.java b/src/main/java/ovh/sad/animalrp/animals/Cow.java new file mode 100644 index 0000000..b92958e --- /dev/null +++ b/src/main/java/ovh/sad/animalrp/animals/Cow.java @@ -0,0 +1,63 @@ +package ovh.sad.animalrp.animals; + +import java.util.Random; + +import net.fabricmc.fabric.api.entity.event.v1.ServerLivingEntityEvents; +import net.minecraft.entity.effect.StatusEffectInstance; +import net.minecraft.entity.effect.StatusEffects; +import net.minecraft.entity.player.PlayerEntity; +import net.minecraft.item.Items; +import net.minecraft.server.network.ServerPlayerEntity; +import net.minecraft.sound.SoundEvents; +import ovh.sad.animalrp.AnimalRP; +import ovh.sad.animalrp.util.Mood; +import ovh.sad.animalrp.util.TextDestroyer; + +public class Cow extends Animal { + TextDestroyer destroyer = new TextDestroyer(new String[] { + "Moo!", "Moooo~", + "Moo?", "Mooooo!", "MooMoo" + }, new String[][] { + { "you", "moo" }, + { "o", "oo" }, + { "i", "ii" }, + { "a", "aa" }, + { "e", "ee" } + }); + Random rand = new Random(); + + public Cow() { + super("cow", "Moo!", "#a52a2a"); + this.moodSounds.put(Mood.HAPPY, SoundEvents.ENTITY_COW_AMBIENT); + this.moodSounds.put(Mood.CUTE, SoundEvents.ENTITY_COW_STEP); + this.moodSounds.put(Mood.SAD, SoundEvents.ENTITY_COW_HURT); + this.moodSounds.put(Mood.STRESSED, SoundEvents.ENTITY_COW_HURT); + this.moodSounds.put(Mood.ANGRY, SoundEvents.ENTITY_COW_DEATH); + + this.superfoods.add(Items.APPLE); + this.superfoods.add(Items.GOLDEN_APPLE); + this.superfoods.add(Items.CARROT); + + ServerLivingEntityEvents.AFTER_DAMAGE.register((entity, source, baseDmg, dmg, blocked) -> { + if (!(source.getSource() instanceof ServerPlayerEntity damaged)) return; + if (entity instanceof PlayerEntity) return; + + Animal animal = AnimalRP.users.get(damaged.getUuid()); + + if (animal == null || !animal.name.equals(this.name)) { + return; + } + + if (rand.nextFloat() < 0.3) { + entity.removeStatusEffect(StatusEffects.SLOWNESS); + entity.addStatusEffect( + new StatusEffectInstance(StatusEffects.SLOWNESS, 20, 2, true, true, true)); + } + }); + } + + @Override + public String chatTransformations(String message) { + return this.destroyer.destroy(message); + } +} diff --git a/src/main/java/ovh/sad/animalrp/animals/Dog.java b/src/main/java/ovh/sad/animalrp/animals/Dog.java index dfd7eab..7e78e23 100644 --- a/src/main/java/ovh/sad/animalrp/animals/Dog.java +++ b/src/main/java/ovh/sad/animalrp/animals/Dog.java @@ -34,7 +34,7 @@ public class Dog extends Animal { if (animal == null) return ActionResult.PASS; - if (animal.name != this.name) + if (!animal.name.equals(this.name)) return ActionResult.PASS; player.removeStatusEffect(StatusEffects.SPEED); player.addStatusEffect( diff --git a/src/main/java/ovh/sad/animalrp/animals/Fox.java b/src/main/java/ovh/sad/animalrp/animals/Fox.java index 5d45e41..c000626 100644 --- a/src/main/java/ovh/sad/animalrp/animals/Fox.java +++ b/src/main/java/ovh/sad/animalrp/animals/Fox.java @@ -42,19 +42,16 @@ public class Fox extends Animal { this.superfoods.add(Items.GLOW_BERRIES); ServerLivingEntityEvents.AFTER_DAMAGE.register((entity, source, baseDmg, dmg, blocked) -> { - if(!(source.getSource() instanceof ServerPlayerEntity)) return; + if(!(source.getSource() instanceof ServerPlayerEntity damager)) return; if(entity instanceof PlayerEntity) return; - LivingEntity victim = entity; - ServerPlayerEntity damager = (ServerPlayerEntity) source.getSource(); - Animal animal = AnimalRP.users.get(damager.getUuid()); if (animal == null || !animal.name.equals(this.name)) { return; } - victim.damage((ServerWorld)victim.getWorld(), victim.getDamageSources().playerAttack(damager), dmg * 0.25F); + entity.damage((ServerWorld) entity.getWorld(), entity.getDamageSources().playerAttack(damager), dmg * 0.25F); }); } diff --git a/src/main/java/ovh/sad/animalrp/commands/InfoCommand.java b/src/main/java/ovh/sad/animalrp/commands/InfoCommand.java index 48949bd..10628a5 100644 --- a/src/main/java/ovh/sad/animalrp/commands/InfoCommand.java +++ b/src/main/java/ovh/sad/animalrp/commands/InfoCommand.java @@ -5,28 +5,29 @@ import java.util.UUID; import com.mojang.brigadier.CommandDispatcher; import eu.pb4.placeholders.api.TextParserUtils; +import eu.pb4.placeholders.api.parsers.TagParser; import net.minecraft.command.CommandRegistryAccess; import net.minecraft.server.command.CommandManager; import net.minecraft.server.command.ServerCommandSource; import net.minecraft.text.Text; import ovh.sad.animalrp.AnimalRP; +import ovh.sad.animalrp.animals.Animal; import ovh.sad.animalrp.util.HashmapStore; public class InfoCommand { public void Command(CommandDispatcher dispatcher, CommandRegistryAccess registryAccess, CommandManager.RegistrationEnvironment environment) { dispatcher.register(CommandManager.literal("rpinfo").executes(context -> { - context.getSource().sendFeedback(() -> TextParserUtils.formatText("AnimalRPs is a mod that adds animals to Minecraft."), false); - context.getSource().sendFeedback(() -> TextParserUtils.formatText("Usage of this mod is very simple."), false); - context.getSource().sendFeedback(() -> TextParserUtils.formatText("- To select a animal, use: /tf"), false); - context.getSource().sendFeedback(() -> TextParserUtils.formatText("- To turn off animals, use: /tf off"), false); - context.getSource().sendFeedback(() -> TextParserUtils.formatText("- Don't want the chat changes? Use: /disableanimalchat."), false); - context.getSource().sendFeedback(() -> Text.empty(), false); - context.getSource().sendFeedback(() -> TextParserUtils.formatText("- To see what the animals do, visit: https://git.sad.ovh/sophie/animalrpfabric"), false); - context.getSource().sendFeedback(() -> TextParserUtils.formatText("- To pet someone, rightclick on them (they must be a animal!)"), false); - context.getSource().sendFeedback(() -> Text.empty(), false); - context.getSource().sendFeedback(() -> TextParserUtils.formatText("Available animals: " + String.join(", ", AnimalRP.animals.keySet().stream().toList())), false); - + context.getSource().sendFeedback(() -> AnimalRP.parseText("AnimalRPs is a mod that adds animals to Minecraft."), false); + context.getSource().sendFeedback(() -> AnimalRP.parseText("Usage of this mod is very simple."), false); + context.getSource().sendFeedback(() -> AnimalRP.parseText("- To select a animal, use: /tf"), false); + context.getSource().sendFeedback(() -> AnimalRP.parseText("- To turn off animals, use: /tf off"), false); + context.getSource().sendFeedback(() -> AnimalRP.parseText("- Don't want the chat changes? Use: /disableanimalchat."), false); + context.getSource().sendFeedback(Text::empty, false); + context.getSource().sendFeedback(() -> AnimalRP.parseText("- To see what the animals do, visit: https://git.sad.ovh/sophie/animalrpfabric"), false); + context.getSource().sendFeedback(() -> AnimalRP.parseText("- To pet someone, rightclick on them (they must be a animal!)"), false); + context.getSource().sendFeedback(Text::empty, false); + context.getSource().sendFeedback(() -> AnimalRP.parseText("To see available animals, run /tf"), false); return 0; })); } diff --git a/src/main/java/ovh/sad/animalrp/commands/InteractionCommand.java b/src/main/java/ovh/sad/animalrp/commands/InteractionCommand.java index cb7443f..5129bce 100644 --- a/src/main/java/ovh/sad/animalrp/commands/InteractionCommand.java +++ b/src/main/java/ovh/sad/animalrp/commands/InteractionCommand.java @@ -2,7 +2,6 @@ package ovh.sad.animalrp.commands; import com.mojang.brigadier.CommandDispatcher; -import eu.pb4.placeholders.api.TextParserUtils; import net.minecraft.command.CommandRegistryAccess; import net.minecraft.command.argument.EntityArgumentType; import net.minecraft.entity.Entity; @@ -10,7 +9,6 @@ import net.minecraft.server.command.CommandManager; import net.minecraft.server.command.ServerCommandSource; import net.minecraft.server.network.ServerPlayerEntity; import net.minecraft.sound.SoundCategory; -import net.minecraft.text.Text; import ovh.sad.animalrp.AnimalRP; import ovh.sad.animalrp.animals.Animal; import ovh.sad.animalrp.util.Mood; @@ -28,7 +26,6 @@ public class InteractionCommand { this.mood = mood; } - @SuppressWarnings("deprecation") public void Command(CommandDispatcher dispatcher, CommandRegistryAccess registryAccess, CommandManager.RegistrationEnvironment environment) { dispatcher.register(CommandManager.literal(this.command) @@ -38,18 +35,17 @@ public class InteractionCommand { if (!(sender instanceof ServerPlayerEntity)) { context.getSource().sendFeedback( - () -> Text.literal("I'm sorry console.").withColor(8421504), false); + () -> AnimalRP.parseText("I'm sorry console."), false); return 0; } - ServerPlayerEntity player = context.getSource().getPlayer(); + ServerPlayerEntity player = context.getSource().getPlayerOrThrow(); Animal aplayer = AnimalRP.users.get(player.getUuid()); if (aplayer == null) { context.getSource().sendFeedback( - () -> Text.literal("Only animals can interact with other animals :(") - .withColor(8421504), + () -> AnimalRP.parseText("Only animals can interact with other animals :("), false); return 0; } @@ -58,8 +54,7 @@ public class InteractionCommand { if (splayer.getName() == player.getName()) { context.getSource().sendFeedback( - () -> Text.literal("You can't " + this.command + " yourself.") - .withColor(8421504), + () -> AnimalRP.parseText("You can't " + this.command + " yourself."), false); return 0; } @@ -67,19 +62,18 @@ public class InteractionCommand { if (asplayer == null) { context.getSource().sendFeedback( - () -> Text.literal(splayer.getName() + " is not an animal! :(") - .withColor(8421504), + () -> AnimalRP.parseText(""+splayer.getName() + " is not an animal! :("), false); return 0; } - splayer.sendMessage(TextParserUtils.formatText( + splayer.sendMessage(AnimalRP.parseText( String.format(this.toThem, - "" + player.getName().getString() + "", + "" + player.getName().getString() + "", "" + aplayer.catchphrase))); - player.sendMessage(TextParserUtils.formatText( + player.sendMessage(AnimalRP.parseText( String.format(this.toYou, - "" + splayer.getName().getString() + "", + "" + splayer.getName().getString() + "", "" + asplayer.catchphrase))); player.getWorld().playSound(splayer, splayer.getBlockPos(), asplayer.moodSounds.get(this.mood), SoundCategory.PLAYERS, 1F, diff --git a/src/main/java/ovh/sad/animalrp/commands/NoChatCommand.java b/src/main/java/ovh/sad/animalrp/commands/NoChatCommand.java index e392c09..b27d0df 100644 --- a/src/main/java/ovh/sad/animalrp/commands/NoChatCommand.java +++ b/src/main/java/ovh/sad/animalrp/commands/NoChatCommand.java @@ -15,19 +15,18 @@ public class NoChatCommand { public void Command(CommandDispatcher dispatcher, CommandRegistryAccess registryAccess, CommandManager.RegistrationEnvironment environment) { dispatcher.register(CommandManager.literal("disableanimalchat").executes(context -> { - UUID userUuid = context.getSource().getEntity().getUuid(); + UUID userUuid = context.getSource().getEntityOrThrow().getUuid(); - Boolean isDisabled = AnimalRP.noChat.get(context.getSource().getEntity().getUuid()); + Boolean isDisabled = AnimalRP.noChat.get(context.getSource().getEntityOrThrow().getUuid()); if (isDisabled == null) isDisabled = false; - if (isDisabled) { // + if (isDisabled) { context.getSource().sendFeedback( - () -> Text.literal("AnimalRP's chat modifications are now enabled for you.").withColor(65280), + () -> AnimalRP.parseText("AnimalRP's chat modifications are now enabled for you."), false); AnimalRP.noChat.remove(userUuid); } else { - context.getSource().sendFeedback(() -> Text - .literal("AnimalRP's chat modifications are now disabled for you.").withColor(16711680), false); + context.getSource().sendFeedback(() -> AnimalRP.parseText("AnimalRP's chat modifications are now disabled for you."), false); AnimalRP.noChat.put(userUuid, true); } HashmapStore.save("nochat.json", AnimalRP.noChat); diff --git a/src/main/java/ovh/sad/animalrp/commands/TfCommand.java b/src/main/java/ovh/sad/animalrp/commands/TfCommand.java index a82cb99..32cdfa5 100644 --- a/src/main/java/ovh/sad/animalrp/commands/TfCommand.java +++ b/src/main/java/ovh/sad/animalrp/commands/TfCommand.java @@ -9,9 +9,6 @@ import net.minecraft.command.CommandRegistryAccess; import net.minecraft.entity.Entity; import net.minecraft.server.command.CommandManager; import net.minecraft.server.command.ServerCommandSource; -import net.minecraft.text.MutableText; -import net.minecraft.text.Text; -import net.minecraft.util.Formatting; import ovh.sad.animalrp.AnimalRP; import ovh.sad.animalrp.animals.Animal; import ovh.sad.animalrp.util.HashmapStore; @@ -20,9 +17,13 @@ public class TfCommand { public void Command(CommandDispatcher dispatcher, CommandRegistryAccess registryAccess, CommandManager.RegistrationEnvironment environment) { dispatcher.register(CommandManager.literal("tf") + .executes(context -> { + classicError(context.getSource(), false); + return 1; + }) .then(CommandManager.argument("animal", StringArgumentType.string()) .executes(context -> { - final Entity entity = context.getSource().getEntity(); + final Entity entity = context.getSource().getEntityOrThrow(); final String animalString = StringArgumentType.getString(context, "animal"); Animal animal = AnimalRP.animals.get(animalString); @@ -30,44 +31,40 @@ public class TfCommand { if (animalString.equals("off")) { if (AnimalRP.users.get(entity.getUuid()) == null) { context.getSource().sendFeedback( - () -> Text.literal("You do not have a animal set."), false); + () -> AnimalRP.parseText("You do not have a animal set."), false); return 0; } AnimalRP.users.remove(entity.getUuid()); context.getSource().sendFeedback( - () -> Text.literal("You no longer have a animal set."), false); + () -> AnimalRP.parseText("You no longer have a animal set."), false); HashmapStore.save("users.json", AnimalRP.users); return 0; } - if (animal == null) { - classicError(context.getSource()); + if(animal == null) { + classicError(context.getSource(), true); return 0; } AnimalRP.users.put(entity.getUuid(), animal); context.getSource() .sendFeedback( - () -> Text.literal("You are now an " + animalString + "! ") - .append(Text.literal(animal.catchphrase) - .formatted(Formatting.ITALIC).withColor( - Integer.parseInt(animal.color.substring(1), - 16))), + () -> AnimalRP.parseText("You are now an " + animalString + "! "+animal.catchphrase), false); HashmapStore.save("users.json", AnimalRP.users); return 1; }))); } - void classicError(ServerCommandSource source) { - MutableText options = Text.literal("Your options are: "); + void classicError(ServerCommandSource source, Boolean showInvalidAnimal) { + StringBuilder options = new StringBuilder("Your options are: "); + for (Entry entry : AnimalRP.animals.entrySet()) { - options.append(Text.literal(entry.getKey() + " ") - .withColor(Integer.parseInt(entry.getValue().color.substring(1), 16))); + options.append(" ").append(entry.getKey()).append(""); } - source.sendFeedback(() -> Text.literal("Invalid animal!").withColor(16711680), false); - source.sendFeedback(() -> options, false); - source.sendFeedback(() -> Text.literal("Use /tf off to disable the changes."), false); + if(showInvalidAnimal) source.sendFeedback(() -> AnimalRP.parseText("Invalid animal!"), false); + source.sendFeedback(() -> AnimalRP.parseText(options.toString()), false); + source.sendFeedback(() -> AnimalRP.parseText("Use /tf off to disable the changes."), false); } } \ No newline at end of file diff --git a/src/main/java/ovh/sad/animalrp/mixin/FoodEating.java b/src/main/java/ovh/sad/animalrp/mixin/FoodEating.java index 560cc8e..ea26245 100644 --- a/src/main/java/ovh/sad/animalrp/mixin/FoodEating.java +++ b/src/main/java/ovh/sad/animalrp/mixin/FoodEating.java @@ -30,7 +30,7 @@ public class FoodEating { Animal animal = AnimalRP.users.get(player.getUuid()); if (animal != null) { - if (animal.name == "bee") { + if (animal.name.equals("bee")) { ((Bee) animal).onEat(player, stack); } if (animal.superfoods.contains(stack.getItem())) {