minecraft-src/net/minecraft/commands/arguments/ResourceOrIdArgument.java
2025-09-18 12:27:44 +00:00

199 lines
8.9 KiB
Java

package net.minecraft.commands.arguments;
import com.mojang.brigadier.ImmutableStringReader;
import com.mojang.brigadier.StringReader;
import com.mojang.brigadier.arguments.ArgumentType;
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.brigadier.suggestion.Suggestions;
import com.mojang.brigadier.suggestion.SuggestionsBuilder;
import com.mojang.serialization.Codec;
import com.mojang.serialization.DynamicOps;
import java.util.Collection;
import java.util.List;
import java.util.Optional;
import java.util.concurrent.CompletableFuture;
import net.minecraft.commands.CommandBuildContext;
import net.minecraft.commands.CommandSourceStack;
import net.minecraft.commands.SharedSuggestionProvider;
import net.minecraft.core.Holder;
import net.minecraft.core.HolderLookup;
import net.minecraft.core.Registry;
import net.minecraft.core.registries.Registries;
import net.minecraft.nbt.NbtOps;
import net.minecraft.nbt.SnbtGrammar;
import net.minecraft.nbt.Tag;
import net.minecraft.network.chat.Component;
import net.minecraft.resources.ResourceKey;
import net.minecraft.resources.ResourceLocation;
import net.minecraft.server.dialog.Dialog;
import net.minecraft.util.parsing.packrat.Atom;
import net.minecraft.util.parsing.packrat.Dictionary;
import net.minecraft.util.parsing.packrat.NamedRule;
import net.minecraft.util.parsing.packrat.Term;
import net.minecraft.util.parsing.packrat.commands.Grammar;
import net.minecraft.util.parsing.packrat.commands.ResourceLocationParseRule;
import net.minecraft.world.level.storage.loot.LootTable;
import net.minecraft.world.level.storage.loot.functions.LootItemFunction;
import net.minecraft.world.level.storage.loot.functions.LootItemFunctions;
import net.minecraft.world.level.storage.loot.predicates.LootItemCondition;
import org.jetbrains.annotations.Nullable;
public class ResourceOrIdArgument<T> implements ArgumentType<Holder<T>> {
private static final Collection<String> EXAMPLES = List.of("foo", "foo:bar", "012", "{}", "true");
public static final DynamicCommandExceptionType ERROR_FAILED_TO_PARSE = new DynamicCommandExceptionType(
object -> Component.translatableEscape("argument.resource_or_id.failed_to_parse", object)
);
public static final Dynamic2CommandExceptionType ERROR_NO_SUCH_ELEMENT = new Dynamic2CommandExceptionType(
(object, object2) -> Component.translatableEscape("argument.resource_or_id.no_such_element", object, object2)
);
public static final DynamicOps<Tag> OPS = NbtOps.INSTANCE;
private final HolderLookup.Provider registryLookup;
private final Optional<? extends HolderLookup.RegistryLookup<T>> elementLookup;
private final Codec<T> codec;
private final Grammar<ResourceOrIdArgument.Result<T, Tag>> grammar;
private final ResourceKey<? extends Registry<T>> registryKey;
protected ResourceOrIdArgument(CommandBuildContext registryLookup, ResourceKey<? extends Registry<T>> registryKey, Codec<T> codec) {
this.registryLookup = registryLookup;
this.elementLookup = registryLookup.lookup(registryKey);
this.registryKey = registryKey;
this.codec = codec;
this.grammar = createGrammar(registryKey, OPS);
}
public static <T, O> Grammar<ResourceOrIdArgument.Result<T, O>> createGrammar(ResourceKey<? extends Registry<T>> registryKey, DynamicOps<O> ops) {
Grammar<O> grammar = SnbtGrammar.createParser(ops);
Dictionary<StringReader> dictionary = new Dictionary<>();
Atom<ResourceOrIdArgument.Result<T, O>> atom = Atom.of("result");
Atom<ResourceLocation> atom2 = Atom.of("id");
Atom<O> atom3 = Atom.of("value");
dictionary.put(atom2, ResourceLocationParseRule.INSTANCE);
dictionary.put(atom3, grammar.top().value());
NamedRule<StringReader, ResourceOrIdArgument.Result<T, O>> namedRule = dictionary.put(
atom, Term.alternative(dictionary.named(atom2), dictionary.named(atom3)), scope -> {
ResourceLocation resourceLocation = scope.get(atom2);
if (resourceLocation != null) {
return new ResourceOrIdArgument.ReferenceResult<>(ResourceKey.create(registryKey, resourceLocation));
} else {
O object = scope.getOrThrow(atom3);
return new ResourceOrIdArgument.InlineResult<>(object);
}
}
);
return new Grammar<>(dictionary, namedRule);
}
public static ResourceOrIdArgument.LootTableArgument lootTable(CommandBuildContext context) {
return new ResourceOrIdArgument.LootTableArgument(context);
}
public static Holder<LootTable> getLootTable(CommandContext<CommandSourceStack> context, String name) throws CommandSyntaxException {
return getResource(context, name);
}
public static ResourceOrIdArgument.LootModifierArgument lootModifier(CommandBuildContext context) {
return new ResourceOrIdArgument.LootModifierArgument(context);
}
public static Holder<LootItemFunction> getLootModifier(CommandContext<CommandSourceStack> context, String name) {
return getResource(context, name);
}
public static ResourceOrIdArgument.LootPredicateArgument lootPredicate(CommandBuildContext context) {
return new ResourceOrIdArgument.LootPredicateArgument(context);
}
public static Holder<LootItemCondition> getLootPredicate(CommandContext<CommandSourceStack> context, String name) {
return getResource(context, name);
}
public static ResourceOrIdArgument.DialogArgument dialog(CommandBuildContext context) {
return new ResourceOrIdArgument.DialogArgument(context);
}
public static Holder<Dialog> getDialog(CommandContext<CommandSourceStack> context, String name) {
return getResource(context, name);
}
private static <T> Holder<T> getResource(CommandContext<CommandSourceStack> context, String name) {
return context.getArgument(name, Holder.class);
}
@Nullable
public Holder<T> parse(StringReader reader) throws CommandSyntaxException {
return this.parse(reader, this.grammar, OPS);
}
@Nullable
private <O> Holder<T> parse(StringReader reader, Grammar<ResourceOrIdArgument.Result<T, O>> grammar, DynamicOps<O> ops) throws CommandSyntaxException {
ResourceOrIdArgument.Result<T, O> result = grammar.parseForCommands(reader);
return this.elementLookup.isEmpty()
? null
: result.parse(reader, this.registryLookup, ops, this.codec, (HolderLookup.RegistryLookup<T>)this.elementLookup.get());
}
@Override
public <S> CompletableFuture<Suggestions> listSuggestions(CommandContext<S> commandContext, SuggestionsBuilder suggestionsBuilder) {
return SharedSuggestionProvider.listSuggestions(commandContext, suggestionsBuilder, this.registryKey, SharedSuggestionProvider.ElementSuggestionType.ELEMENTS);
}
@Override
public Collection<String> getExamples() {
return EXAMPLES;
}
public static class DialogArgument extends ResourceOrIdArgument<Dialog> {
protected DialogArgument(CommandBuildContext context) {
super(context, Registries.DIALOG, Dialog.DIRECT_CODEC);
}
}
public record InlineResult<T, O>(O value) implements ResourceOrIdArgument.Result<T, O> {
@Override
public Holder<T> parse(
ImmutableStringReader reader, HolderLookup.Provider registryLookup, DynamicOps<O> ops, Codec<T> codec, HolderLookup.RegistryLookup<T> elementLookup
) throws CommandSyntaxException {
return Holder.direct(
codec.parse(registryLookup.createSerializationContext(ops), this.value)
.getOrThrow(string -> ResourceOrIdArgument.ERROR_FAILED_TO_PARSE.createWithContext(reader, string))
);
}
}
public static class LootModifierArgument extends ResourceOrIdArgument<LootItemFunction> {
protected LootModifierArgument(CommandBuildContext context) {
super(context, Registries.ITEM_MODIFIER, LootItemFunctions.ROOT_CODEC);
}
}
public static class LootPredicateArgument extends ResourceOrIdArgument<LootItemCondition> {
protected LootPredicateArgument(CommandBuildContext context) {
super(context, Registries.PREDICATE, LootItemCondition.DIRECT_CODEC);
}
}
public static class LootTableArgument extends ResourceOrIdArgument<LootTable> {
protected LootTableArgument(CommandBuildContext context) {
super(context, Registries.LOOT_TABLE, LootTable.DIRECT_CODEC);
}
}
public record ReferenceResult<T, O>(ResourceKey<T> key) implements ResourceOrIdArgument.Result<T, O> {
@Override
public Holder<T> parse(
ImmutableStringReader reader, HolderLookup.Provider registryLookup, DynamicOps<O> ops, Codec<T> codec, HolderLookup.RegistryLookup<T> elementLookup
) throws CommandSyntaxException {
return (Holder<T>)elementLookup.get(this.key)
.orElseThrow(() -> ResourceOrIdArgument.ERROR_NO_SUCH_ELEMENT.createWithContext(reader, this.key.location(), this.key.registry()));
}
}
public sealed interface Result<T, O> permits ResourceOrIdArgument.InlineResult, ResourceOrIdArgument.ReferenceResult {
Holder<T> parse(
ImmutableStringReader reader, HolderLookup.Provider registryLookup, DynamicOps<O> ops, Codec<T> codec, HolderLookup.RegistryLookup<T> elementLookup
) throws CommandSyntaxException;
}
}