Skip to content

Commit c2962b1

Browse files
committed
Initial rewrite of command building system
Precursor work for #483 Notable changes: - No more flattening everything into a CommandAPICommand - CommandTrees are not flattened - CommandMetaData and Execution removed (only used to flatten CommandTrees) - MultiLiteralArgument and subcommands are no longer converted into LiteralArguments - Optional arguments are not chopped up into multiple arrays - In general, a single call to `ExecutableCommand#register` now creates the entire Brigadier tree for that command instance - Should only be converting Arguments to Brigadier nodes once instead of multiple times - CommandAPIHandler no longer creates Brigadier nodes - Delegated to AbstractCommandAPICommand, AbstractCommandTree, AbstractArgumentTree.java, and AbstractArgument - Problems that prevent a command from being registered have been unified under CommandRegistrationException - Exception messages tweaked - Bukkit-specific features removed from `commandapi-core` - Removed `isConverted` property of CommandAPICommand - Moved logic for flattening EntitySelectorArgument from `CommandAPIHandler#generateBrigadierCommand` to Converter - Moved special Bukkit command permission stuff into CommandAPIBukkit - Taking advantage of Brigadier redirects - Command alias nodes redirect to main command name - MultiLiteralArgument literal nodes use redirects - Instead of duplicating the argument node structure after these arguments, they can reference the same path TODO: - The checks performed by `CommandAPIHandler#hasCommandConflict` have not been reimplemented (I'm not sure if they did anything?) - MultiLiteralArgument is currently broken. See Mojang/brigadier#137 - Many changes not covered by tests (could be sneaky behavioral differences)
1 parent d494b7a commit c2962b1

35 files changed

+1873
-1491
lines changed

commandapi-core/src/main/java/dev/jorel/commandapi/AbstractArgumentTree.java

+71-13
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,10 @@
11
package dev.jorel.commandapi;
22

3+
import com.mojang.brigadier.tree.CommandNode;
34
import dev.jorel.commandapi.arguments.AbstractArgument;
5+
import dev.jorel.commandapi.arguments.GreedyArgument;
6+
import dev.jorel.commandapi.exceptions.GreedyArgumentException;
7+
import dev.jorel.commandapi.exceptions.MissingCommandExecutorException;
48

59
import java.util.ArrayList;
610
import java.util.List;
@@ -30,10 +34,10 @@ public abstract class AbstractArgumentTree<Impl
3034
*/
3135
@SuppressWarnings("unchecked")
3236
protected AbstractArgumentTree() {
33-
if (this instanceof AbstractArgument<?, ?, Argument, CommandSender>) {
37+
if (this instanceof AbstractArgument<?, ?, ?, ?>) {
3438
this.argument = (Argument) this;
3539
} else {
36-
throw new IllegalArgumentException("Implicit inherited constructor must be from Argument");
40+
throw new IllegalArgumentException("Implicit inherited constructor must be from AbstractArgument");
3741
}
3842
}
3943

@@ -60,19 +64,73 @@ public Impl then(final AbstractArgumentTree<?, Argument, CommandSender> tree) {
6064
return instance();
6165
}
6266

63-
List<Execution<CommandSender, Argument>> getExecutions() {
64-
List<Execution<CommandSender, Argument>> executions = new ArrayList<>();
65-
// If this is executable, add its execution
66-
if (this.executor.hasAnyExecutors()) {
67-
executions.add(new Execution<>(List.of(this.argument), this.executor));
67+
/**
68+
* Builds the Brigadier {@link CommandNode} structure for this argument tree.
69+
*
70+
* @param previousNode The {@link CommandNode} to add this argument tree onto.
71+
* @param previousArguments A List of CommandAPI arguments that came before this argument tree.
72+
* @param previousNonLiteralArgumentNames A List of Strings containing the node names that came before this argument.
73+
* @param <Source> The Brigadier Source object for running commands.
74+
*/
75+
public <Source> void buildBrigadierNode(CommandNode<Source> previousNode, List<Argument> previousArguments, List<String> previousNonLiteralArgumentNames) {
76+
// Check preconditions
77+
if (argument instanceof GreedyArgument && !arguments.isEmpty()) {
78+
throw new GreedyArgumentException(previousArguments, argument, getBranchesAsList());
79+
}
80+
if (!executor.hasAnyExecutors() && arguments.isEmpty()) {
81+
throw new MissingCommandExecutorException(previousArguments, argument);
82+
}
83+
84+
// Create node for this argument
85+
CommandNode<Source> rootNode = argument.addArgumentNodes(previousNode, previousArguments, previousNonLiteralArgumentNames, executor);
86+
87+
// Add our branches as children to the node
88+
for (AbstractArgumentTree<?, Argument, CommandSender> child : arguments) {
89+
// We need a new list for each branch of the tree
90+
List<Argument> newPreviousArguments = new ArrayList<>(previousArguments);
91+
List<String> newPreviousArgumentNames = new ArrayList<>(previousNonLiteralArgumentNames);
92+
93+
child.buildBrigadierNode(rootNode, newPreviousArguments, newPreviousArgumentNames);
94+
}
95+
}
96+
97+
/**
98+
* @return A list of paths that represent the possible branches of this argument tree as Strings, starting with the
99+
* base argument held by this tree.
100+
*/
101+
public List<List<String>> getBranchesAsStrings() {
102+
List<List<String>> baseArgumentPaths = new ArrayList<>();
103+
baseArgumentPaths.add(new ArrayList<>());
104+
argument.appendToCommandPaths(baseArgumentPaths);
105+
106+
List<List<String>> argumentStrings = new ArrayList<>();
107+
for (AbstractArgumentTree<?, Argument, CommandSender> child : arguments) {
108+
for (List<String> subArgs : child.getBranchesAsStrings()) {
109+
for (List<String> basePath : baseArgumentPaths) {
110+
List<String> mergedPaths = new ArrayList<>();
111+
mergedPaths.addAll(basePath);
112+
mergedPaths.addAll(subArgs);
113+
argumentStrings.add(mergedPaths);
114+
}
115+
}
68116
}
69-
// Add all executions from all arguments
70-
for (AbstractArgumentTree<?, Argument, CommandSender> tree : arguments) {
71-
for (Execution<CommandSender, Argument> execution : tree.getExecutions()) {
72-
// Prepend this argument to the arguments of the executions
73-
executions.add(execution.prependedBy(this.argument));
117+
118+
return argumentStrings;
119+
}
120+
121+
/**
122+
* @return A list of paths that represent the possible branches of this argument tree as Argument objects.
123+
*/
124+
protected List<List<Argument>> getBranchesAsList() {
125+
List<List<Argument>> branchesList = new ArrayList<>();
126+
for (AbstractArgumentTree<?, Argument, CommandSender> branch : arguments) {
127+
for (List<Argument> subBranchList : branch.getBranchesAsList()) {
128+
List<Argument> newBranchList = new ArrayList<>();
129+
newBranchList.add(branch.argument);
130+
newBranchList.addAll(subBranchList);
131+
branchesList.add(newBranchList);
74132
}
75133
}
76-
return executions;
134+
return branchesList;
77135
}
78136
}

0 commit comments

Comments
 (0)