package net.minecraft.advancements.critereon; import com.mojang.serialization.Codec; import com.mojang.serialization.codecs.RecordCodecBuilder; import com.mojang.serialization.codecs.RecordCodecBuilder.Instance; 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 p_43150_, Inventory p_43151_, ItemStack p_43152_) { int i = 0; int j = 0; int k = 0; for (int l = 0; l < p_43151_.getContainerSize(); l++) { ItemStack itemstack = p_43151_.getItem(l); if (itemstack.isEmpty()) { j++; } else { k++; if (itemstack.getCount() >= itemstack.getMaxStackSize()) { i++; } } } this.trigger(p_43150_, p_43151_, p_43152_, i, j, k); } private void trigger(ServerPlayer p_43154_, Inventory p_43155_, ItemStack p_43156_, int p_43157_, int p_43158_, int p_43159_) { this.trigger(p_43154_, p_43166_ -> p_43166_.matches(p_43155_, p_43156_, p_43157_, p_43158_, p_43159_)); } public static record TriggerInstance( Optional player, InventoryChangeTrigger.TriggerInstance.Slots slots, List items ) implements SimpleCriterionTrigger.SimpleInstance { public static final Codec CODEC = RecordCodecBuilder.create( p_325218_ -> p_325218_.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(p_325218_, InventoryChangeTrigger.TriggerInstance::new) ); public static Criterion hasItems(ItemPredicate.Builder... p_297239_) { return hasItems(Stream.of(p_297239_).map(ItemPredicate.Builder::build).toArray(ItemPredicate[]::new)); } public static Criterion hasItems(ItemPredicate... p_43198_) { return CriteriaTriggers.INVENTORY_CHANGED .createCriterion( new InventoryChangeTrigger.TriggerInstance(Optional.empty(), InventoryChangeTrigger.TriggerInstance.Slots.ANY, List.of(p_43198_)) ); } public static Criterion hasItems(ItemLike... p_298202_) { ItemPredicate[] aitempredicate = new ItemPredicate[p_298202_.length]; for (int i = 0; i < p_298202_.length; i++) { aitempredicate[i] = new ItemPredicate( Optional.of(HolderSet.direct(p_298202_[i].asItem().builtInRegistryHolder())), MinMaxBounds.Ints.ANY, DataComponentPredicate.EMPTY, Map.of() ); } return hasItems(aitempredicate); } public boolean matches(Inventory p_43187_, ItemStack p_43188_, int p_43189_, int p_43190_, int p_43191_) { if (!this.slots.matches(p_43189_, p_43190_, p_43191_)) { return false; } else if (this.items.isEmpty()) { return true; } else if (this.items.size() != 1) { List list = new ObjectArrayList<>(this.items); int i = p_43187_.getContainerSize(); for (int j = 0; j < i; j++) { if (list.isEmpty()) { return true; } ItemStack itemstack = p_43187_.getItem(j); if (!itemstack.isEmpty()) { list.removeIf(p_325217_ -> p_325217_.test(itemstack)); } } return list.isEmpty(); } else { return !p_43188_.isEmpty() && this.items.get(0).test(p_43188_); } } @Override public Optional player() { return this.player; } public static record Slots(MinMaxBounds.Ints occupied, MinMaxBounds.Ints full, MinMaxBounds.Ints empty) { public static final Codec CODEC = RecordCodecBuilder.create( p_325219_ -> p_325219_.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(p_325219_, 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 p_310199_, int p_310886_, int p_309638_) { if (!this.full.matches(p_310199_)) { return false; } else { return !this.empty.matches(p_310886_) ? false : this.occupied.matches(p_309638_); } } } } }