minecraft-src/net/minecraft/advancements/critereon/InventoryChangeTrigger.java
2025-07-04 01:41:11 +03:00

131 lines
5.3 KiB
Java

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<InventoryChangeTrigger.TriggerInstance> {
@Override
public Codec<InventoryChangeTrigger.TriggerInstance> 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<ContextAwarePredicate> player, InventoryChangeTrigger.TriggerInstance.Slots slots, List<ItemPredicate> items)
implements SimpleCriterionTrigger.SimpleInstance {
public static final Codec<InventoryChangeTrigger.TriggerInstance> 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<InventoryChangeTrigger.TriggerInstance> hasItems(ItemPredicate.Builder... items) {
return hasItems((ItemPredicate[])Stream.of(items).map(ItemPredicate.Builder::build).toArray(ItemPredicate[]::new));
}
public static Criterion<InventoryChangeTrigger.TriggerInstance> hasItems(ItemPredicate... items) {
return CriteriaTriggers.INVENTORY_CHANGED
.createCriterion(new InventoryChangeTrigger.TriggerInstance(Optional.empty(), InventoryChangeTrigger.TriggerInstance.Slots.ANY, List.of(items)));
}
public static Criterion<InventoryChangeTrigger.TriggerInstance> 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<ItemPredicate> 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<InventoryChangeTrigger.TriggerInstance.Slots> 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);
}
}
}
}
}