minecraft-src/net/minecraft/commands/arguments/item/ItemPredicateArgument.java
2025-07-04 03:45:38 +03:00

225 lines
11 KiB
Java

package net.minecraft.commands.arguments.item;
import com.mojang.brigadier.ImmutableStringReader;
import com.mojang.brigadier.context.CommandContext;
import com.mojang.brigadier.exceptions.CommandSyntaxException;
import com.mojang.brigadier.exceptions.Dynamic2CommandExceptionType;
import com.mojang.brigadier.exceptions.DynamicCommandExceptionType;
import com.mojang.serialization.Codec;
import com.mojang.serialization.DataResult;
import com.mojang.serialization.Decoder;
import com.mojang.serialization.Dynamic;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.function.Predicate;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import net.minecraft.Util;
import net.minecraft.advancements.critereon.MinMaxBounds;
import net.minecraft.commands.CommandBuildContext;
import net.minecraft.commands.CommandSourceStack;
import net.minecraft.core.Holder;
import net.minecraft.core.HolderLookup;
import net.minecraft.core.HolderSet;
import net.minecraft.core.Holder.Reference;
import net.minecraft.core.component.DataComponentType;
import net.minecraft.core.component.predicates.DataComponentPredicate;
import net.minecraft.core.registries.Registries;
import net.minecraft.network.chat.Component;
import net.minecraft.resources.RegistryOps;
import net.minecraft.resources.ResourceKey;
import net.minecraft.resources.ResourceLocation;
import net.minecraft.tags.TagKey;
import net.minecraft.util.parsing.packrat.commands.ParserBasedArgument;
import net.minecraft.world.item.Item;
import net.minecraft.world.item.ItemStack;
public class ItemPredicateArgument extends ParserBasedArgument<ItemPredicateArgument.Result> {
private static final Collection<String> EXAMPLES = Arrays.asList("stick", "minecraft:stick", "#stick", "#stick{foo:'bar'}");
static final DynamicCommandExceptionType ERROR_UNKNOWN_ITEM = new DynamicCommandExceptionType(
object -> Component.translatableEscape("argument.item.id.invalid", object)
);
static final DynamicCommandExceptionType ERROR_UNKNOWN_TAG = new DynamicCommandExceptionType(
object -> Component.translatableEscape("arguments.item.tag.unknown", object)
);
static final DynamicCommandExceptionType ERROR_UNKNOWN_COMPONENT = new DynamicCommandExceptionType(
object -> Component.translatableEscape("arguments.item.component.unknown", object)
);
static final Dynamic2CommandExceptionType ERROR_MALFORMED_COMPONENT = new Dynamic2CommandExceptionType(
(object, object2) -> Component.translatableEscape("arguments.item.component.malformed", object, object2)
);
static final DynamicCommandExceptionType ERROR_UNKNOWN_PREDICATE = new DynamicCommandExceptionType(
object -> Component.translatableEscape("arguments.item.predicate.unknown", object)
);
static final Dynamic2CommandExceptionType ERROR_MALFORMED_PREDICATE = new Dynamic2CommandExceptionType(
(object, object2) -> Component.translatableEscape("arguments.item.predicate.malformed", object, object2)
);
private static final ResourceLocation COUNT_ID = ResourceLocation.withDefaultNamespace("count");
static final Map<ResourceLocation, ItemPredicateArgument.ComponentWrapper> PSEUDO_COMPONENTS = (Map<ResourceLocation, ItemPredicateArgument.ComponentWrapper>)Stream.of(
new ItemPredicateArgument.ComponentWrapper(COUNT_ID, itemStack -> true, MinMaxBounds.Ints.CODEC.map(ints -> itemStack -> ints.matches(itemStack.getCount())))
)
.collect(Collectors.toUnmodifiableMap(ItemPredicateArgument.ComponentWrapper::id, componentWrapper -> componentWrapper));
static final Map<ResourceLocation, ItemPredicateArgument.PredicateWrapper> PSEUDO_PREDICATES = (Map<ResourceLocation, ItemPredicateArgument.PredicateWrapper>)Stream.of(
new ItemPredicateArgument.PredicateWrapper(COUNT_ID, MinMaxBounds.Ints.CODEC.map(ints -> itemStack -> ints.matches(itemStack.getCount())))
)
.collect(Collectors.toUnmodifiableMap(ItemPredicateArgument.PredicateWrapper::id, predicateWrapper -> predicateWrapper));
public ItemPredicateArgument(CommandBuildContext context) {
super(ComponentPredicateParser.createGrammar(new ItemPredicateArgument.Context(context)).mapResult(list -> Util.allOf(list)::test));
}
public static ItemPredicateArgument itemPredicate(CommandBuildContext context) {
return new ItemPredicateArgument(context);
}
public static ItemPredicateArgument.Result getItemPredicate(CommandContext<CommandSourceStack> context, String name) {
return context.getArgument(name, ItemPredicateArgument.Result.class);
}
@Override
public Collection<String> getExamples() {
return EXAMPLES;
}
record ComponentWrapper(ResourceLocation id, Predicate<ItemStack> presenceChecker, Decoder<? extends Predicate<ItemStack>> valueChecker) {
public static <T> ItemPredicateArgument.ComponentWrapper create(ImmutableStringReader reader, ResourceLocation id, DataComponentType<T> componentType) throws CommandSyntaxException {
Codec<T> codec = componentType.codec();
if (codec == null) {
throw ItemPredicateArgument.ERROR_UNKNOWN_COMPONENT.createWithContext(reader, id);
} else {
return new ItemPredicateArgument.ComponentWrapper(id, itemStack -> itemStack.has(componentType), codec.map(object -> itemStack -> {
T object2 = itemStack.get(componentType);
return Objects.equals(object, object2);
}));
}
}
public Predicate<ItemStack> decode(ImmutableStringReader reader, Dynamic<?> data) throws CommandSyntaxException {
DataResult<? extends Predicate<ItemStack>> dataResult = this.valueChecker.parse(data);
return (Predicate<ItemStack>)dataResult.getOrThrow(
string -> ItemPredicateArgument.ERROR_MALFORMED_COMPONENT.createWithContext(reader, this.id.toString(), string)
);
}
}
static class Context
implements ComponentPredicateParser.Context<Predicate<ItemStack>, ItemPredicateArgument.ComponentWrapper, ItemPredicateArgument.PredicateWrapper> {
private final HolderLookup.Provider registries;
private final HolderLookup.RegistryLookup<Item> items;
private final HolderLookup.RegistryLookup<DataComponentType<?>> components;
private final HolderLookup.RegistryLookup<DataComponentPredicate.Type<?>> predicates;
Context(HolderLookup.Provider registries) {
this.registries = registries;
this.items = registries.lookupOrThrow(Registries.ITEM);
this.components = registries.lookupOrThrow(Registries.DATA_COMPONENT_TYPE);
this.predicates = registries.lookupOrThrow(Registries.DATA_COMPONENT_PREDICATE_TYPE);
}
public Predicate<ItemStack> forElementType(ImmutableStringReader immutableStringReader, ResourceLocation resourceLocation) throws CommandSyntaxException {
Reference<Item> reference = (Reference<Item>)this.items
.get(ResourceKey.create(Registries.ITEM, resourceLocation))
.orElseThrow(() -> ItemPredicateArgument.ERROR_UNKNOWN_ITEM.createWithContext(immutableStringReader, resourceLocation));
return itemStack -> itemStack.is(reference);
}
public Predicate<ItemStack> forTagType(ImmutableStringReader immutableStringReader, ResourceLocation resourceLocation) throws CommandSyntaxException {
HolderSet<Item> holderSet = (HolderSet<Item>)this.items
.get(TagKey.create(Registries.ITEM, resourceLocation))
.orElseThrow(() -> ItemPredicateArgument.ERROR_UNKNOWN_TAG.createWithContext(immutableStringReader, resourceLocation));
return itemStack -> itemStack.is(holderSet);
}
public ItemPredicateArgument.ComponentWrapper lookupComponentType(ImmutableStringReader immutableStringReader, ResourceLocation resourceLocation) throws CommandSyntaxException {
ItemPredicateArgument.ComponentWrapper componentWrapper = (ItemPredicateArgument.ComponentWrapper)ItemPredicateArgument.PSEUDO_COMPONENTS
.get(resourceLocation);
if (componentWrapper != null) {
return componentWrapper;
} else {
DataComponentType<?> dataComponentType = (DataComponentType<?>)this.components
.get(ResourceKey.create(Registries.DATA_COMPONENT_TYPE, resourceLocation))
.map(Holder::value)
.orElseThrow(() -> ItemPredicateArgument.ERROR_UNKNOWN_COMPONENT.createWithContext(immutableStringReader, resourceLocation));
return ItemPredicateArgument.ComponentWrapper.create(immutableStringReader, resourceLocation, dataComponentType);
}
}
public Predicate<ItemStack> createComponentTest(
ImmutableStringReader immutableStringReader, ItemPredicateArgument.ComponentWrapper componentWrapper, Dynamic<?> dynamic
) throws CommandSyntaxException {
return componentWrapper.decode(immutableStringReader, RegistryOps.injectRegistryContext(dynamic, this.registries));
}
public Predicate<ItemStack> createComponentTest(ImmutableStringReader immutableStringReader, ItemPredicateArgument.ComponentWrapper componentWrapper) {
return componentWrapper.presenceChecker;
}
public ItemPredicateArgument.PredicateWrapper lookupPredicateType(ImmutableStringReader immutableStringReader, ResourceLocation resourceLocation) throws CommandSyntaxException {
ItemPredicateArgument.PredicateWrapper predicateWrapper = (ItemPredicateArgument.PredicateWrapper)ItemPredicateArgument.PSEUDO_PREDICATES
.get(resourceLocation);
return predicateWrapper != null
? predicateWrapper
: (ItemPredicateArgument.PredicateWrapper)this.predicates
.get(ResourceKey.create(Registries.DATA_COMPONENT_PREDICATE_TYPE, resourceLocation))
.map(ItemPredicateArgument.PredicateWrapper::new)
.orElseThrow(() -> ItemPredicateArgument.ERROR_UNKNOWN_PREDICATE.createWithContext(immutableStringReader, resourceLocation));
}
public Predicate<ItemStack> createPredicateTest(
ImmutableStringReader immutableStringReader, ItemPredicateArgument.PredicateWrapper predicateWrapper, Dynamic<?> dynamic
) throws CommandSyntaxException {
return predicateWrapper.decode(immutableStringReader, RegistryOps.injectRegistryContext(dynamic, this.registries));
}
@Override
public Stream<ResourceLocation> listElementTypes() {
return this.items.listElementIds().map(ResourceKey::location);
}
@Override
public Stream<ResourceLocation> listTagTypes() {
return this.items.listTagIds().map(TagKey::location);
}
@Override
public Stream<ResourceLocation> listComponentTypes() {
return Stream.concat(
ItemPredicateArgument.PSEUDO_COMPONENTS.keySet().stream(),
this.components.listElements().filter(reference -> !((DataComponentType)reference.value()).isTransient()).map(reference -> reference.key().location())
);
}
@Override
public Stream<ResourceLocation> listPredicateTypes() {
return Stream.concat(ItemPredicateArgument.PSEUDO_PREDICATES.keySet().stream(), this.predicates.listElementIds().map(ResourceKey::location));
}
public Predicate<ItemStack> negate(Predicate<ItemStack> predicate) {
return predicate.negate();
}
public Predicate<ItemStack> anyOf(List<Predicate<ItemStack>> list) {
return Util.anyOf(list);
}
}
record PredicateWrapper(ResourceLocation id, Decoder<? extends Predicate<ItemStack>> type) {
public PredicateWrapper(Reference<DataComponentPredicate.Type<?>> predicate) {
this(predicate.key().location(), predicate.value().codec().map(dataComponentPredicate -> dataComponentPredicate::matches));
}
public Predicate<ItemStack> decode(ImmutableStringReader reader, Dynamic<?> data) throws CommandSyntaxException {
DataResult<? extends Predicate<ItemStack>> dataResult = this.type.parse(data);
return (Predicate<ItemStack>)dataResult.getOrThrow(
string -> ItemPredicateArgument.ERROR_MALFORMED_PREDICATE.createWithContext(reader, this.id.toString(), string)
);
}
}
public interface Result extends Predicate<ItemStack> {
}
}