218 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
			
		
		
	
	
			218 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.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 reader, ResourceLocation elementType) throws CommandSyntaxException {
 | |
| 			Holder.Reference<Item> reference = (Holder.Reference<Item>)this.items
 | |
| 				.get(ResourceKey.create(Registries.ITEM, elementType))
 | |
| 				.orElseThrow(() -> ItemPredicateArgument.ERROR_UNKNOWN_ITEM.createWithContext(reader, elementType));
 | |
| 			return itemStack -> itemStack.is(reference);
 | |
| 		}
 | |
| 
 | |
| 		public Predicate<ItemStack> forTagType(ImmutableStringReader reader, ResourceLocation tagType) throws CommandSyntaxException {
 | |
| 			HolderSet<Item> holderSet = (HolderSet<Item>)this.items
 | |
| 				.get(TagKey.create(Registries.ITEM, tagType))
 | |
| 				.orElseThrow(() -> ItemPredicateArgument.ERROR_UNKNOWN_TAG.createWithContext(reader, tagType));
 | |
| 			return itemStack -> itemStack.is(holderSet);
 | |
| 		}
 | |
| 
 | |
| 		public ItemPredicateArgument.ComponentWrapper lookupComponentType(ImmutableStringReader reader, ResourceLocation componentType) throws CommandSyntaxException {
 | |
| 			ItemPredicateArgument.ComponentWrapper componentWrapper = (ItemPredicateArgument.ComponentWrapper)ItemPredicateArgument.PSEUDO_COMPONENTS.get(componentType);
 | |
| 			if (componentWrapper != null) {
 | |
| 				return componentWrapper;
 | |
| 			} else {
 | |
| 				DataComponentType<?> dataComponentType = (DataComponentType<?>)this.components
 | |
| 					.get(ResourceKey.create(Registries.DATA_COMPONENT_TYPE, componentType))
 | |
| 					.map(Holder::value)
 | |
| 					.orElseThrow(() -> ItemPredicateArgument.ERROR_UNKNOWN_COMPONENT.createWithContext(reader, componentType));
 | |
| 				return ItemPredicateArgument.ComponentWrapper.create(reader, componentType, dataComponentType);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		public Predicate<ItemStack> createComponentTest(ImmutableStringReader reader, ItemPredicateArgument.ComponentWrapper context, Dynamic<?> data) throws CommandSyntaxException {
 | |
| 			return context.decode(reader, RegistryOps.injectRegistryContext(data, this.registries));
 | |
| 		}
 | |
| 
 | |
| 		public Predicate<ItemStack> createComponentTest(ImmutableStringReader reader, ItemPredicateArgument.ComponentWrapper context) {
 | |
| 			return context.presenceChecker;
 | |
| 		}
 | |
| 
 | |
| 		public ItemPredicateArgument.PredicateWrapper lookupPredicateType(ImmutableStringReader reader, ResourceLocation predicateType) throws CommandSyntaxException {
 | |
| 			ItemPredicateArgument.PredicateWrapper predicateWrapper = (ItemPredicateArgument.PredicateWrapper)ItemPredicateArgument.PSEUDO_PREDICATES.get(predicateType);
 | |
| 			return predicateWrapper != null
 | |
| 				? predicateWrapper
 | |
| 				: (ItemPredicateArgument.PredicateWrapper)this.predicates
 | |
| 					.get(ResourceKey.create(Registries.DATA_COMPONENT_PREDICATE_TYPE, predicateType))
 | |
| 					.map(ItemPredicateArgument.PredicateWrapper::new)
 | |
| 					.orElseThrow(() -> ItemPredicateArgument.ERROR_UNKNOWN_PREDICATE.createWithContext(reader, predicateType));
 | |
| 		}
 | |
| 
 | |
| 		public Predicate<ItemStack> createPredicateTest(ImmutableStringReader reader, ItemPredicateArgument.PredicateWrapper predicateType, Dynamic<?> data) throws CommandSyntaxException {
 | |
| 			return predicateType.decode(reader, RegistryOps.injectRegistryContext(data, 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> value) {
 | |
| 			return value.negate();
 | |
| 		}
 | |
| 
 | |
| 		public Predicate<ItemStack> anyOf(List<Predicate<ItemStack>> values) {
 | |
| 			return Util.anyOf(values);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	record PredicateWrapper(ResourceLocation id, Decoder<? extends Predicate<ItemStack>> type) {
 | |
| 		public PredicateWrapper(Holder.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> {
 | |
| 	}
 | |
| }
 |