SteamWar/SpigotCore
Archiviert
13
0

Add CommandNode

Start complete rework of CommandSystem
Dieser Commit ist enthalten in:
yoyosource 2021-07-09 13:38:24 +02:00
Ursprung bc966247a8
Commit 5240ca3bf4
3 geänderte Dateien mit 165 neuen und 34 gelöschten Zeilen

Datei anzeigen

@ -0,0 +1,93 @@
/*
* This file is a part of the SteamWar software.
*
* Copyright (C) 2020 SteamWar.de-Serverteam
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
package de.steamwar.command;
import org.bukkit.command.CommandSender;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.stream.Collectors;
class CommandNode {
private final TypeMapper<?> typeMapper;
private boolean varArg = false;
private List<CommandNode> commandNodeList = new ArrayList<>();
private Method executor;
CommandNode(TypeMapper<?> typeMapper) {
this.typeMapper = typeMapper;
}
public void addNode(CommandNode commandNode) {
commandNodeList.add(commandNode);
this.varArg = false;
}
public void setExecutor(Method method) {
this.executor = executor;
}
public void setVarArg(boolean varArg) {
if (commandNodeList.isEmpty()) {
this.varArg = varArg;
} else {
this.varArg = false;
}
}
public List<String> tabComplete(CommandSender commandSender, int index, String[] args) {
try {
if (index == args.length - 1) {
return typeMapper.tabCompletes(commandSender, Arrays.copyOf(args, index), args[args.length - 1]);
}
if (typeMapper.map(commandSender, Arrays.copyOf(args, index), args[index]) == null) {
return Collections.emptyList();
}
if (varArg) {
return tabComplete(commandSender, index + 1, args);
} else {
return commandNodeList.stream()
.map(commandNode -> commandNode.tabComplete(commandSender, index + 1, args))
.flatMap(List::stream)
.collect(Collectors.toList());
}
} catch (Exception e) {
return Collections.emptyList();
}
}
public boolean execute(CommandSender commandSender, int index, String[] args) {
if (varArg) {
for (int i = index; i < args.length; i++) {
String[] previousArgs = Arrays.copyOf(args, i);
typeMapper.map(commandSender, previousArgs, args[i]);
}
} else {
}
// if (args.length)
return false;
}
}

Datei anzeigen

@ -23,6 +23,7 @@ import com.mojang.brigadier.arguments.*;
import com.mojang.brigadier.builder.ArgumentBuilder; import com.mojang.brigadier.builder.ArgumentBuilder;
import com.mojang.brigadier.builder.LiteralArgumentBuilder; import com.mojang.brigadier.builder.LiteralArgumentBuilder;
import com.mojang.brigadier.builder.RequiredArgumentBuilder; import com.mojang.brigadier.builder.RequiredArgumentBuilder;
import com.mojang.brigadier.tree.CommandNode;
import net.minecraft.server.v1_15_R1.CommandListenerWrapper; import net.minecraft.server.v1_15_R1.CommandListenerWrapper;
import org.bukkit.command.CommandSender; import org.bukkit.command.CommandSender;
@ -60,51 +61,88 @@ class SWCommandBrigadier implements SWCommand.SWCommandInterface {
private SubCommand createSubCommand(Method method, String[] strings) { private SubCommand createSubCommand(Method method, String[] strings) {
return new SubCommand(swCommand, method, strings, localTypeMapper, current -> { return new SubCommand(swCommand, method, strings, localTypeMapper, current -> {
ArgumentBuilder argumentBuilder = null; List<ArgumentBuilder> argumentBuilders = new ArrayList<>();
for (String s : current.subCommand) { for (String s : current.subCommand) {
LiteralArgumentBuilder literalArgumentBuilder = LiteralArgumentBuilder.literal(s); LiteralArgumentBuilder literalArgumentBuilder = LiteralArgumentBuilder.literal(s);
if (argumentBuilder != null) { List<ArgumentBuilder> currentBuilders = new ArrayList<>();
argumentBuilder.then(literalArgumentBuilder); argumentBuilders.forEach(currrentBuilder -> {
} else { currentBuilders.add(currrentBuilder.then(literalArgumentBuilder));
current.argumentNode = literalArgumentBuilder; });
if (argumentBuilders.isEmpty()) {
currentBuilders.add(literalArgumentBuilder);
if (current.argumentNode.isEmpty()) {
current.argumentNode.add(literalArgumentBuilder);
}
} }
argumentBuilder = literalArgumentBuilder; argumentBuilders = currentBuilders;
} }
for (int i = 0; i < current.arguments.length - (current.varArgType != null ? 1 : 0); i++) { for (int i = 0; i < current.arguments.length - (current.varArgType != null ? 1 : 0); i++) {
Parameter parameter = current.parameters[i + 1]; Parameter parameter = current.parameters[i + 1];
Class<?> parameterType = parameter.getType(); Class<?> parameterType = parameter.getType();
ArgumentType<?> argumentType = getArgumentType(parameter, parameterType, current); ArgumentType<?> argumentType = getArgumentType(parameter, parameterType);
RequiredArgumentBuilder requiredArgumentBuilder = RequiredArgumentBuilder.argument(parameter.getName(), argumentType); List<ArgumentBuilder> newBuilders = new ArrayList<>();
if (argumentBuilder != null) { if (argumentType == null) {
argumentBuilder.then(requiredArgumentBuilder); try {
current.arguments[i].tabCompletes(null, null, "").stream()
.map(LiteralArgumentBuilder::literal)
.forEach(newBuilders::add);
} catch (Exception e) {
newBuilders.add(RequiredArgumentBuilder.argument(parameter.getName(), StringArgumentType.string()));
}
} else { } else {
current.argumentNode = requiredArgumentBuilder; newBuilders.add(RequiredArgumentBuilder.argument(parameter.getName(), argumentType));
} }
argumentBuilder = requiredArgumentBuilder;
List<ArgumentBuilder> currentBuilders = new ArrayList<>();
argumentBuilders.forEach(currrentBuilder -> {
newBuilders.forEach(argumentBuilder -> {
currentBuilders.add(currrentBuilder.then(argumentBuilder));
});
});
if (argumentBuilders.isEmpty()) {
currentBuilders.addAll(newBuilders);
if (current.argumentNode.isEmpty()) {
current.argumentNode.addAll(newBuilders);
}
}
argumentBuilders = currentBuilders;
} }
if (current.varArgType != null) { if (current.varArgType != null) {
// TODO: UNSUPPORTED Parameter parameter = current.parameters[current.parameters.length - 1];
/*Parameter parameter = parameters[parameters.length - 1];
Class<?> parameterType = parameter.getType(); Class<?> parameterType = parameter.getType();
TypeMapper<?> typeMapper = arguments[arguments.length - 1]; ArgumentType<?> argumentType = getArgumentType(parameter, parameterType);
ArgumentType<?> argumentType = getArgumentType(parameter, parameterType, typeMapper); List<ArgumentBuilder> newBuilders = new ArrayList<>();
if (argumentType == null) {
RequiredArgumentBuilder requiredArgumentBuilder = RequiredArgumentBuilder.argument(parameter.getName(), argumentType); try {
if (argumentBuilder != null) { current.arguments[current.arguments.length - 1].tabCompletes(null, null, "").stream()
argumentBuilder.then(requiredArgumentBuilder); .map(LiteralArgumentBuilder::literal)
.forEach(newBuilders::add);
} catch (Exception e) {
newBuilders.add(RequiredArgumentBuilder.argument(parameter.getName(), StringArgumentType.string()));
}
} else { } else {
argumentNode = requiredArgumentBuilder; newBuilders.add(RequiredArgumentBuilder.argument(parameter.getName(), argumentType));
} }
argumentBuilder = requiredArgumentBuilder;
argumentBuilder.executes(commandContext -> { // TODO: VarArgs
invoke((CommandSender) commandContext.getCommand(), commandContext.getInput().split(" ")); List<ArgumentBuilder> currentBuilders = new ArrayList<>();
return 0; argumentBuilders.forEach(currrentBuilder -> {
newBuilders.forEach(argumentBuilder -> {
currentBuilders.add(currrentBuilder.then(argumentBuilder));
});
}); });
CommandNode<?> commandNode = argumentBuilder.build(); if (argumentBuilders.isEmpty()) {
argumentBuilder.redirect(commandNode);*/ currentBuilders.addAll(newBuilders);
if (current.argumentNode.isEmpty()) {
current.argumentNode.addAll(newBuilders);
}
}
currentBuilders.forEach(argumentBuilder -> argumentBuilder.executes(executes(current)));
argumentBuilders = currentBuilders;
} }
current.argumentNodeEnd = argumentBuilder; current.argumentNodeEnd.addAll(argumentBuilders);
}); });
} }
@ -126,8 +164,8 @@ class SWCommandBrigadier implements SWCommand.SWCommandInterface {
SWCommand.dispatcher.register(literalArgumentBuilder); SWCommand.dispatcher.register(literalArgumentBuilder);
return; return;
} }
subCommand.argumentNodeEnd.executes(executes(subCommand)); subCommand.argumentNodeEnd.forEach(argumentBuilder -> argumentBuilder.executes(executes(subCommand)));
literalArgumentBuilder.then(subCommand.argumentNode); subCommand.argumentNode.forEach(literalArgumentBuilder::then);
SWCommand.dispatcher.register(literalArgumentBuilder); SWCommand.dispatcher.register(literalArgumentBuilder);
} }
@ -147,7 +185,7 @@ class SWCommandBrigadier implements SWCommand.SWCommandInterface {
}); });
} }
private ArgumentType<?> getArgumentType(Parameter parameter, Class<?> parameterType, SubCommand subCommand) { private ArgumentType<?> getArgumentType(Parameter parameter, Class<?> parameterType) {
ArgumentType<?> argumentType; ArgumentType<?> argumentType;
if (parameterType == boolean.class || parameterType == Boolean.class) { if (parameterType == boolean.class || parameterType == Boolean.class) {
argumentType = BoolArgumentType.bool(); argumentType = BoolArgumentType.bool();
@ -168,7 +206,7 @@ class SWCommandBrigadier implements SWCommand.SWCommandInterface {
if (doubleRange != null) argumentType = DoubleArgumentType.doubleArg(doubleRange.min(), doubleRange.max()); if (doubleRange != null) argumentType = DoubleArgumentType.doubleArg(doubleRange.min(), doubleRange.max());
else argumentType = DoubleArgumentType.doubleArg(); else argumentType = DoubleArgumentType.doubleArg();
} else { } else {
argumentType = StringArgumentType.string(); return null;
} }
return argumentType; return argumentType;
} }

Datei anzeigen

@ -34,8 +34,8 @@ import static de.steamwar.command.SWCommandUtils.*;
class SubCommand { class SubCommand {
ArgumentBuilder argumentNode = null; List<ArgumentBuilder> argumentNode = new ArrayList<>();
ArgumentBuilder argumentNodeEnd = null; List<ArgumentBuilder> argumentNodeEnd = new ArrayList<>();
SWCommand swCommand; SWCommand swCommand;
Parameter[] parameters; Parameter[] parameters;