bukkit: Use WrappedBrigadierParser for modern ItemStack suggestions
This commit is contained in:
parent
56f8b58489
commit
99040ca68a
8 changed files with 158 additions and 33 deletions
|
|
@ -4,5 +4,5 @@
|
|||
"https://checkstyle.org/dtds/suppressions_1_2.dtd">
|
||||
<suppressions>
|
||||
<suppress checks="(?:(?:Member|Method)Name|DesignForExtension|Javadoc.*)" files=".*[\\/]mixin[\\/].*"/>
|
||||
<suppress checks="(?:Javadoc.*)" files=".*[\\/]CraftBukkitReflection.java"/>
|
||||
<suppress checks="(?:Javadoc.*)" files=".*[\\/]bukkit[\\/]internal[\\/].*"/>
|
||||
</suppressions>
|
||||
|
|
|
|||
|
|
@ -23,9 +23,13 @@
|
|||
//
|
||||
package cloud.commandframework.bukkit;
|
||||
|
||||
import cloud.commandframework.brigadier.argument.WrappedBrigadierParser;
|
||||
import cloud.commandframework.bukkit.internal.BukkitBackwardsBrigadierSenderMapper;
|
||||
import cloud.commandframework.bukkit.internal.CraftBukkitReflection;
|
||||
import cloud.commandframework.execution.preprocessor.CommandPreprocessingContext;
|
||||
import cloud.commandframework.execution.preprocessor.CommandPreprocessor;
|
||||
import org.checkerframework.checker.nullness.qual.NonNull;
|
||||
import org.checkerframework.checker.nullness.qual.Nullable;
|
||||
|
||||
import java.util.Set;
|
||||
|
||||
|
|
@ -39,6 +43,7 @@ final class BukkitCommandPreprocessor<C> implements CommandPreprocessor<C> {
|
|||
|
||||
private final BukkitCommandManager<C> commandManager;
|
||||
private final Set<CloudBukkitCapabilities> bukkitCapabilities;
|
||||
private final @Nullable BukkitBackwardsBrigadierSenderMapper<C, ?> mapper;
|
||||
|
||||
/**
|
||||
* The Bukkit Command Preprocessor for storing Bukkit-specific contexts in the command contexts
|
||||
|
|
@ -48,10 +53,25 @@ final class BukkitCommandPreprocessor<C> implements CommandPreprocessor<C> {
|
|||
BukkitCommandPreprocessor(final @NonNull BukkitCommandManager<C> commandManager) {
|
||||
this.commandManager = commandManager;
|
||||
this.bukkitCapabilities = commandManager.queryCapabilities();
|
||||
if (this.bukkitCapabilities.contains(CloudBukkitCapabilities.BRIGADIER) && CraftBukkitReflection.craftBukkit()) {
|
||||
this.mapper = new BukkitBackwardsBrigadierSenderMapper<>(this.commandManager);
|
||||
} else {
|
||||
this.mapper = null;
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void accept(final @NonNull CommandPreprocessingContext<C> context) {
|
||||
if (this.mapper != null) {
|
||||
// If the server is Brigadier capable but the Brigadier manager has not been registered, store the native
|
||||
// sender in context manually so that getting suggestions from WrappedBrigadierParser works like expected.
|
||||
if (!context.getCommandContext().contains(WrappedBrigadierParser.COMMAND_CONTEXT_BRIGADIER_NATIVE_SENDER)) {
|
||||
context.getCommandContext().store(
|
||||
WrappedBrigadierParser.COMMAND_CONTEXT_BRIGADIER_NATIVE_SENDER,
|
||||
this.mapper.apply(context.getCommandContext().getSender())
|
||||
);
|
||||
}
|
||||
}
|
||||
context.getCommandContext().store(
|
||||
BukkitCommandContextKeys.BUKKIT_COMMAND_SENDER,
|
||||
this.commandManager.getBackwardsCommandSenderMapper().apply(context.getCommandContext().getSender())
|
||||
|
|
|
|||
|
|
@ -25,6 +25,7 @@ package cloud.commandframework.bukkit;
|
|||
|
||||
import cloud.commandframework.Command;
|
||||
import cloud.commandframework.brigadier.CloudBrigadierManager;
|
||||
import cloud.commandframework.bukkit.internal.BukkitBackwardsBrigadierSenderMapper;
|
||||
import cloud.commandframework.bukkit.internal.CraftBukkitReflection;
|
||||
import cloud.commandframework.context.CommandContext;
|
||||
import com.mojang.brigadier.tree.CommandNode;
|
||||
|
|
@ -36,7 +37,6 @@ import org.bukkit.command.CommandSender;
|
|||
import org.checkerframework.checker.nullness.qual.NonNull;
|
||||
import org.checkerframework.checker.nullness.qual.Nullable;
|
||||
|
||||
import java.lang.reflect.Method;
|
||||
import java.util.Collections;
|
||||
|
||||
@SuppressWarnings({"unchecked", "rawtypes"})
|
||||
|
|
@ -64,19 +64,9 @@ class CloudCommodoreManager<C> extends BukkitPluginRegistrationHandler<C> {
|
|||
|
||||
new BukkitBrigadierMapper<>(this.commandManager, this.brigadierManager);
|
||||
|
||||
if (!CraftBukkitReflection.craftBukkit()) {
|
||||
return;
|
||||
if (CraftBukkitReflection.craftBukkit()) {
|
||||
this.brigadierManager.backwardsBrigadierSenderMapper(new BukkitBackwardsBrigadierSenderMapper<>(this.commandManager));
|
||||
}
|
||||
final Class<?> vanillaCommandWrapperClass = CraftBukkitReflection.needOBCClass("command.VanillaCommandWrapper");
|
||||
final Method getListenerMethod = CraftBukkitReflection.needMethod(
|
||||
vanillaCommandWrapperClass, "getListener", CommandSender.class);
|
||||
this.brigadierManager.backwardsBrigadierSenderMapper(cloud -> {
|
||||
try {
|
||||
return getListenerMethod.invoke(null, this.commandManager.getBackwardsCommandSenderMapper().apply(cloud));
|
||||
} catch (final ReflectiveOperationException e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@Override
|
||||
|
|
|
|||
|
|
@ -0,0 +1,61 @@
|
|||
//
|
||||
// MIT License
|
||||
//
|
||||
// Copyright (c) 2021 Alexander Söderberg & Contributors
|
||||
//
|
||||
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
// of this software and associated documentation files (the "Software"), to deal
|
||||
// in the Software without restriction, including without limitation the rights
|
||||
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
// copies of the Software, and to permit persons to whom the Software is
|
||||
// furnished to do so, subject to the following conditions:
|
||||
//
|
||||
// The above copyright notice and this permission notice shall be included in all
|
||||
// copies or substantial portions of the Software.
|
||||
//
|
||||
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
// SOFTWARE.
|
||||
//
|
||||
package cloud.commandframework.bukkit.internal;
|
||||
|
||||
import cloud.commandframework.bukkit.BukkitCommandManager;
|
||||
import com.google.common.annotations.Beta;
|
||||
import org.bukkit.command.CommandSender;
|
||||
import org.checkerframework.checker.nullness.qual.NonNull;
|
||||
|
||||
import java.lang.reflect.Method;
|
||||
import java.util.function.Function;
|
||||
|
||||
/**
|
||||
* This is not API, and as such, may break, change, or be removed without any notice.
|
||||
*/
|
||||
@Beta
|
||||
public final class BukkitBackwardsBrigadierSenderMapper<C, S> implements Function<C, S> {
|
||||
|
||||
private static final Class<?> VANILLA_COMMAND_WRAPPER_CLASS =
|
||||
CraftBukkitReflection.needOBCClass("command.VanillaCommandWrapper");
|
||||
private static final Method GET_LISTENER_METHOD =
|
||||
CraftBukkitReflection.needMethod(VANILLA_COMMAND_WRAPPER_CLASS, "getListener", CommandSender.class);
|
||||
|
||||
private final BukkitCommandManager<C> commandManager;
|
||||
|
||||
public BukkitBackwardsBrigadierSenderMapper(final @NonNull BukkitCommandManager<C> commandManager) {
|
||||
this.commandManager = commandManager;
|
||||
}
|
||||
|
||||
@SuppressWarnings("unchecked")
|
||||
@Override
|
||||
public S apply(final @NonNull C cloud) {
|
||||
try {
|
||||
return (S) GET_LISTENER_METHOD.invoke(null, this.commandManager.getBackwardsCommandSenderMapper().apply(cloud));
|
||||
} catch (final ReflectiveOperationException e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
|
@ -0,0 +1,29 @@
|
|||
//
|
||||
// MIT License
|
||||
//
|
||||
// Copyright (c) 2021 Alexander Söderberg & Contributors
|
||||
//
|
||||
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
// of this software and associated documentation files (the "Software"), to deal
|
||||
// in the Software without restriction, including without limitation the rights
|
||||
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
// copies of the Software, and to permit persons to whom the Software is
|
||||
// furnished to do so, subject to the following conditions:
|
||||
//
|
||||
// The above copyright notice and this permission notice shall be included in all
|
||||
// copies or substantial portions of the Software.
|
||||
//
|
||||
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
// SOFTWARE.
|
||||
//
|
||||
/**
|
||||
* Internal classes for the cloud-bukkit implementation. Classes in this package
|
||||
* are not part of the API, and as such, may break, change, or be removed without
|
||||
* any notice.
|
||||
*/
|
||||
package cloud.commandframework.bukkit.internal;
|
||||
|
|
@ -173,10 +173,7 @@ public final class ItemStackArgument<C> extends CommandArgument<C, ProtoItemStac
|
|||
final @NonNull CommandContext<C> commandContext,
|
||||
final @NonNull String input
|
||||
) {
|
||||
return Arrays.stream(Material.values())
|
||||
.filter(Material::isItem)
|
||||
.map(value -> value.name().toLowerCase(Locale.ROOT))
|
||||
.collect(Collectors.toList());
|
||||
return this.parser.suggestions(commandContext, input);
|
||||
}
|
||||
|
||||
}
|
||||
|
|
@ -228,6 +225,14 @@ public final class ItemStackArgument<C> extends CommandArgument<C, ProtoItemStac
|
|||
return this.parser.parse(commandContext, inputQueue);
|
||||
}
|
||||
|
||||
@Override
|
||||
public @NonNull List<@NonNull String> suggestions(
|
||||
final @NonNull CommandContext<C> commandContext,
|
||||
final @NonNull String input
|
||||
) {
|
||||
return this.parser.suggestions(commandContext, input);
|
||||
}
|
||||
|
||||
private static final class ModernProtoItemStack implements ProtoItemStack {
|
||||
|
||||
private final Object itemInput;
|
||||
|
|
@ -294,6 +299,17 @@ public final class ItemStackArgument<C> extends CommandArgument<C, ProtoItemStac
|
|||
return this.parser.parse(commandContext, inputQueue);
|
||||
}
|
||||
|
||||
@Override
|
||||
public @NonNull List<@NonNull String> suggestions(
|
||||
final @NonNull CommandContext<C> commandContext,
|
||||
final @NonNull String input
|
||||
) {
|
||||
return Arrays.stream(Material.values())
|
||||
.filter(Material::isItem)
|
||||
.map(value -> value.name().toLowerCase(Locale.ROOT))
|
||||
.collect(Collectors.toList());
|
||||
}
|
||||
|
||||
private static final class LegacyProtoItemStack implements ProtoItemStack {
|
||||
|
||||
private final Material material;
|
||||
|
|
|
|||
|
|
@ -27,18 +27,17 @@ import cloud.commandframework.CommandTree;
|
|||
import cloud.commandframework.arguments.CommandArgument;
|
||||
import cloud.commandframework.brigadier.CloudBrigadierManager;
|
||||
import cloud.commandframework.bukkit.BukkitBrigadierMapper;
|
||||
import cloud.commandframework.bukkit.internal.BukkitBackwardsBrigadierSenderMapper;
|
||||
import cloud.commandframework.bukkit.internal.CraftBukkitReflection;
|
||||
import cloud.commandframework.context.CommandContext;
|
||||
import cloud.commandframework.permission.CommandPermission;
|
||||
import com.destroystokyo.paper.brigadier.BukkitBrigadierCommandSource;
|
||||
import org.bukkit.Bukkit;
|
||||
import org.bukkit.command.CommandSender;
|
||||
import org.bukkit.command.PluginIdentifiableCommand;
|
||||
import org.bukkit.event.EventHandler;
|
||||
import org.bukkit.event.Listener;
|
||||
import org.checkerframework.checker.nullness.qual.NonNull;
|
||||
|
||||
import java.lang.reflect.Method;
|
||||
import java.util.function.BiPredicate;
|
||||
import java.util.regex.Pattern;
|
||||
|
||||
|
|
@ -59,20 +58,10 @@ class PaperBrigadierListener<C> implements Listener {
|
|||
|
||||
new BukkitBrigadierMapper<>(this.paperCommandManager, this.brigadierManager);
|
||||
|
||||
if (!CraftBukkitReflection.craftBukkit()) {
|
||||
return;
|
||||
if (CraftBukkitReflection.craftBukkit()) {
|
||||
this.brigadierManager
|
||||
.backwardsBrigadierSenderMapper(new BukkitBackwardsBrigadierSenderMapper<>(this.paperCommandManager));
|
||||
}
|
||||
final Class<?> vanillaCommandWrapperClass = CraftBukkitReflection.needOBCClass("command.VanillaCommandWrapper");
|
||||
final Method getListenerMethod = CraftBukkitReflection.needMethod(
|
||||
vanillaCommandWrapperClass, "getListener", CommandSender.class);
|
||||
this.brigadierManager.backwardsBrigadierSenderMapper(cloud -> {
|
||||
try {
|
||||
return (BukkitBrigadierCommandSource) getListenerMethod
|
||||
.invoke(null, this.paperCommandManager.getBackwardsCommandSenderMapper().apply(cloud));
|
||||
} catch (final ReflectiveOperationException e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
protected @NonNull CloudBrigadierManager<C, BukkitBrigadierCommandSource> brigadierManager() {
|
||||
|
|
|
|||
|
|
@ -26,8 +26,11 @@ package cloud.commandframework.examples.bukkit;
|
|||
import cloud.commandframework.ArgumentDescription;
|
||||
import cloud.commandframework.Command;
|
||||
import cloud.commandframework.CommandTree;
|
||||
import cloud.commandframework.arguments.compound.ArgumentPair;
|
||||
import cloud.commandframework.bukkit.data.ProtoItemStack;
|
||||
import cloud.commandframework.keys.SimpleCloudKey;
|
||||
import cloud.commandframework.minecraft.extras.MinecraftExceptionHandler;
|
||||
import cloud.commandframework.minecraft.extras.MinecraftExtrasMetaKeys;
|
||||
import cloud.commandframework.minecraft.extras.MinecraftHelp;
|
||||
import cloud.commandframework.annotations.AnnotationParser;
|
||||
import cloud.commandframework.annotations.Argument;
|
||||
|
|
@ -62,6 +65,7 @@ import cloud.commandframework.minecraft.extras.TextColorArgument;
|
|||
import cloud.commandframework.paper.PaperCommandManager;
|
||||
import cloud.commandframework.permission.PredicatePermission;
|
||||
import cloud.commandframework.tasks.TaskConsumer;
|
||||
import cloud.commandframework.types.tuples.Pair;
|
||||
import cloud.commandframework.types.tuples.Triplet;
|
||||
import io.leangen.geantyref.TypeToken;
|
||||
import net.kyori.adventure.identity.Identity;
|
||||
|
|
@ -410,6 +414,22 @@ public final class ExamplePlugin extends JavaPlugin {
|
|||
(sender, key) -> "'{input}' is not very cash money of you"
|
||||
);
|
||||
}
|
||||
|
||||
// vanilla-like give command
|
||||
this.manager.command(this.manager.commandBuilder("givetest")
|
||||
.senderType(Player.class)
|
||||
.argument(ArgumentPair.of(
|
||||
this.manager,
|
||||
"itemstack",
|
||||
Pair.of("item", "amount"),
|
||||
Pair.of(ProtoItemStack.class, Integer.class)
|
||||
).withMapper(ItemStack.class, (sender, pair) -> {
|
||||
final ProtoItemStack proto = pair.getFirst();
|
||||
final int amount = pair.getSecond();
|
||||
return proto.createItemStack(amount, true);
|
||||
}), ArgumentDescription.of("The ItemStack to give"))
|
||||
.meta(MinecraftExtrasMetaKeys.DESCRIPTION, text("Vanilla-like give command"))
|
||||
.handler(ctx -> ((Player) ctx.getSender()).getInventory().addItem(ctx.get("itemstack"))));
|
||||
}
|
||||
|
||||
@CommandMethod("example help [query]")
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue