package net.minecraft.advancements.critereon; import com.google.common.collect.Lists; import com.google.common.collect.Sets; import com.mojang.serialization.Codec; import com.mojang.serialization.codecs.RecordCodecBuilder; import java.util.Collection; import java.util.Iterator; import java.util.List; import java.util.Optional; import java.util.Set; import net.minecraft.advancements.CriteriaTriggers; import net.minecraft.advancements.Criterion; import net.minecraft.advancements.critereon.MinMaxBounds.Ints; import net.minecraft.advancements.critereon.SimpleCriterionTrigger.SimpleInstance; import net.minecraft.core.HolderGetter; import net.minecraft.server.level.ServerPlayer; import net.minecraft.world.entity.Entity; import net.minecraft.world.entity.EntityType; import net.minecraft.world.item.Item; import net.minecraft.world.item.ItemStack; import net.minecraft.world.item.Items; import net.minecraft.world.level.storage.loot.LootContext; import org.jetbrains.annotations.Nullable; public class KilledByArrowTrigger extends SimpleCriterionTrigger { @Override public Codec codec() { return KilledByArrowTrigger.TriggerInstance.CODEC; } public void trigger(ServerPlayer player, Collection victims, @Nullable ItemStack firedFromWeapon) { List list = Lists.newArrayList(); Set> set = Sets.>newHashSet(); for (Entity entity : victims) { set.add(entity.getType()); list.add(EntityPredicate.createContext(player, entity)); } this.trigger(player, triggerInstance -> triggerInstance.matches(list, set.size(), firedFromWeapon)); } public record TriggerInstance( Optional player, List victims, Ints uniqueEntityTypes, Optional firedFromWeapon ) implements SimpleInstance { public static final Codec CODEC = RecordCodecBuilder.create( instance -> instance.group( EntityPredicate.ADVANCEMENT_CODEC.optionalFieldOf("player").forGetter(KilledByArrowTrigger.TriggerInstance::player), EntityPredicate.ADVANCEMENT_CODEC.listOf().optionalFieldOf("victims", List.of()).forGetter(KilledByArrowTrigger.TriggerInstance::victims), Ints.CODEC.optionalFieldOf("unique_entity_types", Ints.ANY).forGetter(KilledByArrowTrigger.TriggerInstance::uniqueEntityTypes), ItemPredicate.CODEC.optionalFieldOf("fired_from_weapon").forGetter(KilledByArrowTrigger.TriggerInstance::firedFromWeapon) ) .apply(instance, KilledByArrowTrigger.TriggerInstance::new) ); public static Criterion crossbowKilled(HolderGetter itemRegistry, EntityPredicate.Builder... victims) { return CriteriaTriggers.KILLED_BY_ARROW .createCriterion( new KilledByArrowTrigger.TriggerInstance( Optional.empty(), EntityPredicate.wrap(victims), Ints.ANY, Optional.of(ItemPredicate.Builder.item().of(itemRegistry, Items.CROSSBOW).build()) ) ); } public static Criterion crossbowKilled(HolderGetter itemRegistry, Ints uniqueEntityTypes) { return CriteriaTriggers.KILLED_BY_ARROW .createCriterion( new KilledByArrowTrigger.TriggerInstance( Optional.empty(), List.of(), uniqueEntityTypes, Optional.of(ItemPredicate.Builder.item().of(itemRegistry, Items.CROSSBOW).build()) ) ); } public boolean matches(Collection context, int uniqueEntityTypes, @Nullable ItemStack firedFromWeapon) { if (!this.firedFromWeapon.isPresent() || firedFromWeapon != null && ((ItemPredicate)this.firedFromWeapon.get()).test(firedFromWeapon)) { if (!this.victims.isEmpty()) { List list = Lists.newArrayList(context); for (ContextAwarePredicate contextAwarePredicate : this.victims) { boolean bl = false; Iterator iterator = list.iterator(); while (iterator.hasNext()) { LootContext lootContext = (LootContext)iterator.next(); if (contextAwarePredicate.matches(lootContext)) { iterator.remove(); bl = true; break; } } if (!bl) { return false; } } } return this.uniqueEntityTypes.matches(uniqueEntityTypes); } else { return false; } } @Override public void validate(CriterionValidator validator) { SimpleInstance.super.validate(validator); validator.validateEntities(this.victims, ".victims"); } } }