package net.minecraft.advancements.critereon; import com.mojang.serialization.Codec; import com.mojang.serialization.codecs.RecordCodecBuilder; import it.unimi.dsi.fastutil.objects.ObjectArrayList; import java.util.List; import java.util.Map; import java.util.Optional; import java.util.stream.Stream; import net.minecraft.advancements.CriteriaTriggers; import net.minecraft.advancements.Criterion; import net.minecraft.core.HolderSet; import net.minecraft.core.component.DataComponentPredicate; import net.minecraft.server.level.ServerPlayer; import net.minecraft.world.entity.player.Inventory; import net.minecraft.world.item.ItemStack; import net.minecraft.world.level.ItemLike; public class InventoryChangeTrigger extends SimpleCriterionTrigger { @Override public Codec codec() { return InventoryChangeTrigger.TriggerInstance.CODEC; } public void trigger(ServerPlayer player, Inventory inventory, ItemStack stack) { int i = 0; int j = 0; int k = 0; for (int l = 0; l < inventory.getContainerSize(); l++) { ItemStack itemStack = inventory.getItem(l); if (itemStack.isEmpty()) { j++; } else { k++; if (itemStack.getCount() >= itemStack.getMaxStackSize()) { i++; } } } this.trigger(player, inventory, stack, i, j, k); } private void trigger(ServerPlayer player, Inventory inventory, ItemStack stack, int full, int empty, int occupied) { this.trigger(player, triggerInstance -> triggerInstance.matches(inventory, stack, full, empty, occupied)); } public record TriggerInstance(Optional player, InventoryChangeTrigger.TriggerInstance.Slots slots, List items) implements SimpleCriterionTrigger.SimpleInstance { public static final Codec CODEC = RecordCodecBuilder.create( instance -> instance.group( EntityPredicate.ADVANCEMENT_CODEC.optionalFieldOf("player").forGetter(InventoryChangeTrigger.TriggerInstance::player), InventoryChangeTrigger.TriggerInstance.Slots.CODEC .optionalFieldOf("slots", InventoryChangeTrigger.TriggerInstance.Slots.ANY) .forGetter(InventoryChangeTrigger.TriggerInstance::slots), ItemPredicate.CODEC.listOf().optionalFieldOf("items", List.of()).forGetter(InventoryChangeTrigger.TriggerInstance::items) ) .apply(instance, InventoryChangeTrigger.TriggerInstance::new) ); public static Criterion hasItems(ItemPredicate.Builder... items) { return hasItems((ItemPredicate[])Stream.of(items).map(ItemPredicate.Builder::build).toArray(ItemPredicate[]::new)); } public static Criterion hasItems(ItemPredicate... items) { return CriteriaTriggers.INVENTORY_CHANGED .createCriterion(new InventoryChangeTrigger.TriggerInstance(Optional.empty(), InventoryChangeTrigger.TriggerInstance.Slots.ANY, List.of(items))); } public static Criterion hasItems(ItemLike... items) { ItemPredicate[] itemPredicates = new ItemPredicate[items.length]; for (int i = 0; i < items.length; i++) { itemPredicates[i] = new ItemPredicate( Optional.of(HolderSet.direct(items[i].asItem().builtInRegistryHolder())), MinMaxBounds.Ints.ANY, DataComponentPredicate.EMPTY, Map.of() ); } return hasItems(itemPredicates); } public boolean matches(Inventory inventory, ItemStack stack, int full, int empty, int occupied) { if (!this.slots.matches(full, empty, occupied)) { return false; } else if (this.items.isEmpty()) { return true; } else if (this.items.size() != 1) { List list = new ObjectArrayList<>(this.items); int i = inventory.getContainerSize(); for (int j = 0; j < i; j++) { if (list.isEmpty()) { return true; } ItemStack itemStack = inventory.getItem(j); if (!itemStack.isEmpty()) { list.removeIf(itemPredicate -> itemPredicate.test(itemStack)); } } return list.isEmpty(); } else { return !stack.isEmpty() && ((ItemPredicate)this.items.get(0)).test(stack); } } public record Slots(MinMaxBounds.Ints occupied, MinMaxBounds.Ints full, MinMaxBounds.Ints empty) { public static final Codec CODEC = RecordCodecBuilder.create( instance -> instance.group( MinMaxBounds.Ints.CODEC.optionalFieldOf("occupied", MinMaxBounds.Ints.ANY).forGetter(InventoryChangeTrigger.TriggerInstance.Slots::occupied), MinMaxBounds.Ints.CODEC.optionalFieldOf("full", MinMaxBounds.Ints.ANY).forGetter(InventoryChangeTrigger.TriggerInstance.Slots::full), MinMaxBounds.Ints.CODEC.optionalFieldOf("empty", MinMaxBounds.Ints.ANY).forGetter(InventoryChangeTrigger.TriggerInstance.Slots::empty) ) .apply(instance, InventoryChangeTrigger.TriggerInstance.Slots::new) ); public static final InventoryChangeTrigger.TriggerInstance.Slots ANY = new InventoryChangeTrigger.TriggerInstance.Slots( MinMaxBounds.Ints.ANY, MinMaxBounds.Ints.ANY, MinMaxBounds.Ints.ANY ); public boolean matches(int full, int empty, int occupied) { if (!this.full.matches(full)) { return false; } else { return !this.empty.matches(empty) ? false : this.occupied.matches(occupied); } } } } }