From 14c7734fb134980073422c9d33078da482052cf6 Mon Sep 17 00:00:00 2001 From: Bukkit/Spigot Date: Tue, 9 Oct 2012 14:54:12 -0500 Subject: [PATCH] Add tab-completion API. Fixes BUKKIT-2181. Adds BUKKIT-2602 CommandMap contains a method that will auto-complete commands appropriately. Before the first space, it searches for commands of which the sender has permission. After the first space, it delegates to the individual command. Vanilla commands contain implementations to mimic vanilla implementation. Exception would be give, that allows for name matching; a feature we already allowed as part of the command is now supported for auto-complete as well. Plugin commands can get a tab completer set to delegate the completion for. If no tab completer is set, it can check the executor to see if it implements the tab completion interface. It will also attempt to chain calls if null gets returned from these interfaces. Plugins also implement the new TabCompleter interface, to add ease-of-use for plugin developers, similar to the onCommand() method. The default command implementation simply searches for player names. To help facilitate command completion, a utility class was added with two functions. One checks two strings, to see if the specified string starts with (ignoring case) the second. The other method uses the first to selectively copy elements from one collection to another. By: Score_Under --- .../main/java/org/bukkit/command/Command.java | 54 +++++++++-- .../java/org/bukkit/command/CommandMap.java | 12 +++ .../bukkit/command/MultipleCommandAlias.java | 2 +- .../org/bukkit/command/PluginCommand.java | 67 +++++++++++++ .../org/bukkit/command/SimpleCommandMap.java | 97 +++++++++++++++++-- .../bukkit/command/TabCommandExecutor.java | 4 + .../java/org/bukkit/command/TabCompleter.java | 20 ++++ .../java/org/bukkit/command/TabExecutor.java | 7 ++ .../bukkit/command/defaults/BanCommand.java | 21 +++- .../bukkit/command/defaults/BanIpCommand.java | 21 +++- .../command/defaults/BanListCommand.java | 19 +++- .../command/defaults/BukkitCommand.java | 6 +- .../defaults/DefaultGameModeCommand.java | 22 +++++ .../bukkit/command/defaults/DeopCommand.java | 30 +++++- .../bukkit/command/defaults/ExpCommand.java | 17 ++++ .../command/defaults/GameModeCommand.java | 27 +++++- .../bukkit/command/defaults/GiveCommand.java | 59 +++++++++++ .../bukkit/command/defaults/HelpCommand.java | 31 +++++- .../bukkit/command/defaults/KickCommand.java | 17 ++++ .../bukkit/command/defaults/KillCommand.java | 14 +++ .../bukkit/command/defaults/ListCommand.java | 14 +++ .../bukkit/command/defaults/OpCommand.java | 44 +++++++++ .../command/defaults/PardonCommand.java | 27 ++++++ .../command/defaults/PardonIpCommand.java | 19 ++++ .../command/defaults/PluginsCommand.java | 3 +- .../command/defaults/ReloadCommand.java | 3 +- .../bukkit/command/defaults/SaveCommand.java | 14 +++ .../command/defaults/SaveOffCommand.java | 14 +++ .../command/defaults/SaveOnCommand.java | 14 +++ .../bukkit/command/defaults/SayCommand.java | 16 +++ .../bukkit/command/defaults/SeedCommand.java | 14 +++ .../bukkit/command/defaults/StopCommand.java | 14 +++ .../command/defaults/TeleportCommand.java | 17 ++++ .../bukkit/command/defaults/TimeCommand.java | 24 +++++ .../command/defaults/TimingsCommand.java | 21 +++- .../defaults/ToggleDownfallCommand.java | 14 +++ .../command/defaults/VanillaCommand.java | 4 +- .../command/defaults/VersionCommand.java | 26 ++++- .../command/defaults/WhitelistCommand.java | 41 ++++++++ .../main/java/org/bukkit/plugin/Plugin.java | 4 +- .../org/bukkit/plugin/java/JavaPlugin.java | 7 ++ .../main/java/org/bukkit/util/StringUtil.java | 49 ++++++++++ .../java/org/bukkit/plugin/TestPlugin.java | 5 + 43 files changed, 904 insertions(+), 51 deletions(-) create mode 100644 paper-api/src/main/java/org/bukkit/command/TabCompleter.java create mode 100644 paper-api/src/main/java/org/bukkit/command/TabExecutor.java create mode 100644 paper-api/src/main/java/org/bukkit/util/StringUtil.java diff --git a/paper-api/src/main/java/org/bukkit/command/Command.java b/paper-api/src/main/java/org/bukkit/command/Command.java index ec502af595..b0b7f21f1c 100644 --- a/paper-api/src/main/java/org/bukkit/command/Command.java +++ b/paper-api/src/main/java/org/bukkit/command/Command.java @@ -1,12 +1,19 @@ package org.bukkit.command; import java.util.ArrayList; +import java.util.Collections; import java.util.List; import java.util.Set; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.ChatColor; import org.bukkit.Server; +import org.bukkit.entity.Player; import org.bukkit.permissions.Permissible; +import org.bukkit.util.StringUtil; + +import com.google.common.collect.ImmutableList; /** * Represents a Command, which executes various tasks upon user input @@ -48,21 +55,48 @@ public abstract class Command { public abstract boolean execute(CommandSender sender, String commandLabel, String[] args); /** - * Executed on tab completion for this command, returning a list of options - * the player can tab through. - *

- * By returning null, you tell Bukkit to generate a list of players to send - * to the sender. - * By returning an empty list, no options will be sent. - * - * @param sender Source object which is executing this command - * @param args All arguments passed to the command, split via ' ' - * @return null to generate a Player list, otherwise a list of options + * @deprecated This method is not supported and returns null */ + @Deprecated public List tabComplete(CommandSender sender, String[] args) { return null; } + /** + * Executed on tab completion for this command, returning a list of options + * the player can tab through. + * + * @param sender Source object which is executing this command + * @param alias the alias being used + * @param args All arguments passed to the command, split via ' ' + * @return a list of tab-completions for the specified arguments. This will never be null. List may be immutable. + * @throws IllegalArgumentException if sender, alias, or args is null + */ + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + if (!(sender instanceof Player) || args.length == 0) { + return ImmutableList.of(); + } + + String lastWord = args[args.length - 1]; + + Player senderPlayer = (Player) sender; + + ArrayList matchedPlayers = new ArrayList(); + for (Player player : sender.getServer().getOnlinePlayers()) { + String name = player.getName(); + if (senderPlayer.canSee(player) && StringUtil.startsWithIgnoreCase(name, lastWord)) { + matchedPlayers.add(name); + } + } + + Collections.sort(matchedPlayers, String.CASE_INSENSITIVE_ORDER); + return matchedPlayers; + } + /** * Returns the name of this command * diff --git a/paper-api/src/main/java/org/bukkit/command/CommandMap.java b/paper-api/src/main/java/org/bukkit/command/CommandMap.java index 5524da2767..e7fad28755 100644 --- a/paper-api/src/main/java/org/bukkit/command/CommandMap.java +++ b/paper-api/src/main/java/org/bukkit/command/CommandMap.java @@ -62,4 +62,16 @@ public interface CommandMap { * @return Command with the specified name or null if a command with that label doesn't exist */ public Command getCommand(String name); + + + /** + * Looks for the requested command and executes an appropriate tab-completer if found. This method will also tab-complete partial commands. + * + * @param sender The command's sender. + * @param cmdLine The entire command string to tab-complete, excluding initial slash. + * @return a list of possible tab-completions. This list may be immutable. Will be null if no matching command of which sender has permission. + * @throws CommandException Thrown when the tab-completer for the given command fails with an unhandled exception + * @throws IllegalArgumentException if either sender or cmdLine are null + */ + public List tabComplete(CommandSender sender, String cmdLine) throws IllegalArgumentException; } diff --git a/paper-api/src/main/java/org/bukkit/command/MultipleCommandAlias.java b/paper-api/src/main/java/org/bukkit/command/MultipleCommandAlias.java index b1b83b1f6b..3a666d16d8 100644 --- a/paper-api/src/main/java/org/bukkit/command/MultipleCommandAlias.java +++ b/paper-api/src/main/java/org/bukkit/command/MultipleCommandAlias.java @@ -10,7 +10,7 @@ public class MultipleCommandAlias extends Command { super(name); this.commands = commands; } - + public Command[] getCommands() { return commands; } diff --git a/paper-api/src/main/java/org/bukkit/command/PluginCommand.java b/paper-api/src/main/java/org/bukkit/command/PluginCommand.java index 05975c7f63..f82e3ed368 100644 --- a/paper-api/src/main/java/org/bukkit/command/PluginCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/PluginCommand.java @@ -1,5 +1,8 @@ package org.bukkit.command; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.plugin.Plugin; /** @@ -8,6 +11,7 @@ import org.bukkit.plugin.Plugin; public final class PluginCommand extends Command implements PluginIdentifiableCommand { private final Plugin owningPlugin; private CommandExecutor executor; + private TabCompleter completer; protected PluginCommand(String name, Plugin owner) { super(name); @@ -69,6 +73,26 @@ public final class PluginCommand extends Command implements PluginIdentifiableCo return executor; } + /** + * Sets the {@link TabCompleter} to run when tab-completing this command. + * If no TabCompleter is specified, and the command's executor implements + * TabCompleter, then the executor will be used for tab completion. + * + * @param completer New tab completer + */ + public void setTabCompleter(TabCompleter completer) { + this.completer = completer; + } + + /** + * Gets the {@link TabCompleter} associated with this command. + * + * @return TabCompleter object linked to this command + */ + public TabCompleter getTabCompleter() { + return completer; + } + /** * Gets the owner of this PluginCommand * @@ -77,4 +101,47 @@ public final class PluginCommand extends Command implements PluginIdentifiableCo public Plugin getPlugin() { return owningPlugin; } + + /** + * {@inheritDoc}
+ *
+ * Delegates to the tab completer if present.
+ * If it is not present or returns null, will delegate to the current command + * executor if it implements {@link TabCompleter}. If a non-null list has not + * been found, will default to standard player name completion in + * {@link Command#tabComplete(CommandSender, String, String[])}.
+ *
+ * This method does not consider permissions. + * @throws CommandException if the completer or executor throw an exception during the process of tab-completing. + * @throws IllegalArgumentException if sender, alias, or args is null + */ + @Override + public java.util.List tabComplete(CommandSender sender, String alias, String[] args) throws CommandException, IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + List completions = null; + try { + if (completer != null) { + completions = completer.onTabComplete(sender, this, alias, args); + } + if (completions == null && executor instanceof TabCompleter) { + completions = ((TabCompleter) executor).onTabComplete(sender, this, alias, args); + } + } catch (Throwable ex) { + StringBuilder message = new StringBuilder(); + message.append("Unhandled exception during tab completion for command '/").append(alias).append(' '); + for (String arg : args) { + message.append(arg).append(' '); + } + message.deleteCharAt(message.length() - 1).append("' in plugin ").append(owningPlugin.getDescription().getFullName()); + throw new CommandException(message.toString(), ex); + } + + if (completions == null) { + return super.tabComplete(sender, alias, args); + } + return completions; + } } diff --git a/paper-api/src/main/java/org/bukkit/command/SimpleCommandMap.java b/paper-api/src/main/java/org/bukkit/command/SimpleCommandMap.java index 384a01ec25..a44504ef1a 100644 --- a/paper-api/src/main/java/org/bukkit/command/SimpleCommandMap.java +++ b/paper-api/src/main/java/org/bukkit/command/SimpleCommandMap.java @@ -1,13 +1,25 @@ package org.bukkit.command; -import org.bukkit.command.defaults.*; - -import java.util.*; - -import org.bukkit.Server; import static org.bukkit.util.Java15Compat.Arrays_copyOfRange; +import java.util.ArrayList; +import java.util.Collection; +import java.util.Collections; +import java.util.HashMap; +import java.util.HashSet; +import java.util.Iterator; +import java.util.List; +import java.util.Map; +import java.util.Set; +import java.util.regex.Pattern; + +import org.apache.commons.lang.Validate; +import org.bukkit.Server; +import org.bukkit.command.defaults.*; +import org.bukkit.util.StringUtil; + public class SimpleCommandMap implements CommandMap { + private static final Pattern PATTERN_ON_SPACE = Pattern.compile(" ", Pattern.LITERAL); protected final Map knownCommands = new HashMap(); protected final Set aliases = new HashSet(); private final Server server; @@ -81,7 +93,7 @@ public class SimpleCommandMap implements CommandMap { Iterator iterator = command.getAliases().iterator(); while (iterator.hasNext()) { - if (!register((String) iterator.next(), fallbackPrefix, command, true)) { + if (!register(iterator.next(), fallbackPrefix, command, true)) { iterator.remove(); } } @@ -150,7 +162,7 @@ public class SimpleCommandMap implements CommandMap { * {@inheritDoc} */ public boolean dispatch(CommandSender sender, String commandLine) throws CommandException { - String[] args = commandLine.split(" "); + String[] args = PATTERN_ON_SPACE.split(commandLine); if (args.length == 0) { return false; @@ -186,13 +198,82 @@ public class SimpleCommandMap implements CommandMap { } public Command getCommand(String name) { - Command target = knownCommands.get(name.toLowerCase()); + Command target = knownCommands.get(name.toLowerCase()); if (target == null) { target = getFallback(name); } return target; } + public List tabComplete(CommandSender sender, String cmdLine) { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(cmdLine, "Command line cannot null"); + + int spaceIndex = cmdLine.indexOf(' '); + + if (spaceIndex == -1) { + ArrayList completions = new ArrayList(); + Map knownCommands = this.knownCommands; + + for (VanillaCommand command : fallbackCommands) { + String name = command.getName(); + + if (!command.testPermissionSilent(sender)) { + continue; + } + if (knownCommands.containsKey(name)) { + // Don't let a vanilla command override a command added below + // This has to do with the way aliases work + continue; + } + if (!StringUtil.startsWithIgnoreCase(name, cmdLine)) { + continue; + } + + completions.add('/' + name); + } + + for (Map.Entry commandEntry : knownCommands.entrySet()) { + Command command = commandEntry.getValue(); + + if (!command.testPermissionSilent(sender)) { + continue; + } + + String name = commandEntry.getKey(); // Use the alias, not command name + + if (StringUtil.startsWithIgnoreCase(name, cmdLine)) { + completions.add('/' + name); + } + } + + Collections.sort(completions, String.CASE_INSENSITIVE_ORDER); + return completions; + } + + String commandName = cmdLine.substring(0, spaceIndex); + Command target = getCommand(commandName); + + if (target == null) { + return null; + } + + if (!target.testPermissionSilent(sender)) { + return null; + } + + String argLine = cmdLine.substring(spaceIndex + 1, cmdLine.length()); + String[] args = PATTERN_ON_SPACE.split(argLine, -1); + + try { + return target.tabComplete(sender, commandName, args); + } catch (CommandException ex) { + throw ex; + } catch (Throwable ex) { + throw new CommandException("Unhandled exception executing tab-completer for '" + cmdLine + "' in " + target, ex); + } + } + public Collection getCommands() { return knownCommands.values(); } diff --git a/paper-api/src/main/java/org/bukkit/command/TabCommandExecutor.java b/paper-api/src/main/java/org/bukkit/command/TabCommandExecutor.java index b2ee58c525..95efbd4089 100644 --- a/paper-api/src/main/java/org/bukkit/command/TabCommandExecutor.java +++ b/paper-api/src/main/java/org/bukkit/command/TabCommandExecutor.java @@ -4,7 +4,11 @@ import java.util.List; /** * Represents a class which can handle command tab completion and commands + * @deprecated Remains for plugins that would have implemented it even without functionality + * @see TabExecutor */ +@Deprecated public interface TabCommandExecutor extends CommandExecutor { public List onTabComplete(); + } diff --git a/paper-api/src/main/java/org/bukkit/command/TabCompleter.java b/paper-api/src/main/java/org/bukkit/command/TabCompleter.java new file mode 100644 index 0000000000..3ba64fe8cb --- /dev/null +++ b/paper-api/src/main/java/org/bukkit/command/TabCompleter.java @@ -0,0 +1,20 @@ +package org.bukkit.command; + +import java.util.List; + +/** + * Represents a class which can suggest tab completions for commands. + */ +public interface TabCompleter { + + /** + * Requests a list of possible completions for a command argument. + * + * @param sender Source of the command + * @param command Command which was executed + * @param alias The alias used + * @param args The arguments passed to the command, including final partial argument to be completed and command label + * @return A List of possible completions for the final argument, or null to default to the command executor + */ + public List onTabComplete(CommandSender sender, Command command, String alias, String[] args); +} diff --git a/paper-api/src/main/java/org/bukkit/command/TabExecutor.java b/paper-api/src/main/java/org/bukkit/command/TabExecutor.java new file mode 100644 index 0000000000..67b3503dbc --- /dev/null +++ b/paper-api/src/main/java/org/bukkit/command/TabExecutor.java @@ -0,0 +1,7 @@ +package org.bukkit.command; + +/** + * This class is provided as a convenience to implement both TabCompleter and CommandExecutor. + */ +public interface TabExecutor extends TabCompleter, CommandExecutor { +} diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/BanCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/BanCommand.java index c4d3a54c73..c436652487 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/BanCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/BanCommand.java @@ -1,12 +1,16 @@ package org.bukkit.command.defaults; import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.ChatColor; import org.bukkit.command.Command; import org.bukkit.command.CommandSender; import org.bukkit.entity.Player; +import com.google.common.collect.ImmutableList; + public class BanCommand extends VanillaCommand { public BanCommand() { super("ban"); @@ -35,13 +39,20 @@ public class BanCommand extends VanillaCommand { return true; } - @Override - public List tabComplete(CommandSender sender, String[] args) { - return args.length >= 1 ? null : EMPTY_LIST; - } - @Override public boolean matches(String input) { return input.equalsIgnoreCase("ban"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + if (args.length >= 1) { + return super.tabComplete(sender, alias, args); + } + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/BanIpCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/BanIpCommand.java index b104973938..8ea0c35685 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/BanIpCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/BanIpCommand.java @@ -2,12 +2,16 @@ package org.bukkit.command.defaults; import java.util.List; import java.util.regex.Pattern; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.ChatColor; import org.bukkit.command.Command; import org.bukkit.command.CommandSender; import org.bukkit.entity.Player; +import com.google.common.collect.ImmutableList; + public class BanIpCommand extends VanillaCommand { public static final Pattern ipValidity = Pattern.compile("^([01]?\\d\\d?|2[0-4]\\d|25[0-5])\\.([01]?\\d\\d?|2[0-4]\\d|25[0-5])\\.([01]?\\d\\d?|2[0-4]\\d|25[0-5])\\.([01]?\\d\\d?|2[0-4]\\d|25[0-5])$"); @@ -43,11 +47,6 @@ public class BanIpCommand extends VanillaCommand { return true; } - @Override - public List tabComplete(CommandSender sender, String[] args) { - return args.length >= 1 ? null : EMPTY_LIST; - } - private void processIPBan(String ip, CommandSender sender) { // TODO: Kick on ban Bukkit.banIP(ip); @@ -59,4 +58,16 @@ public class BanIpCommand extends VanillaCommand { public boolean matches(String input) { return input.equalsIgnoreCase("ban-ip"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + if (args.length == 1) { + return super.tabComplete(sender, alias, args); + } + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/BanListCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/BanListCommand.java index 668a487737..08c5be9a48 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/BanListCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/BanListCommand.java @@ -1,11 +1,19 @@ package org.bukkit.command.defaults; +import java.util.ArrayList; import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.OfflinePlayer; import org.bukkit.command.CommandSender; +import org.bukkit.util.StringUtil; + +import com.google.common.collect.ImmutableList; public class BanListCommand extends VanillaCommand { + private static final List BANLIST_TYPES = ImmutableList.of("ips", "players"); + public BanListCommand() { super("banlist"); this.description = "View all players banned from this server"; @@ -38,8 +46,15 @@ public class BanListCommand extends VanillaCommand { } @Override - public List tabComplete(CommandSender sender, String[] args) { - return args.length >= 1 ? null : EMPTY_LIST; + public List tabComplete(CommandSender sender, String alias, String[] args) { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + if (args.length == 1) { + return StringUtil.copyPartialMatches(args[0], BANLIST_TYPES, new ArrayList(BANLIST_TYPES.size())); + } + return ImmutableList.of(); } @Override diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/BukkitCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/BukkitCommand.java index d3de7c8bfe..23c8580091 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/BukkitCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/BukkitCommand.java @@ -1,10 +1,10 @@ package org.bukkit.command.defaults; -import org.bukkit.command.Command; - import java.util.List; -public abstract class BukkitCommand extends Command{ +import org.bukkit.command.Command; + +public abstract class BukkitCommand extends Command { protected BukkitCommand(String name) { super(name); } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/DefaultGameModeCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/DefaultGameModeCommand.java index 63390b1591..1611d413e6 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/DefaultGameModeCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/DefaultGameModeCommand.java @@ -1,11 +1,20 @@ package org.bukkit.command.defaults; +import java.util.ArrayList; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.GameMode; import org.bukkit.command.Command; import org.bukkit.command.CommandSender; +import org.bukkit.util.StringUtil; + +import com.google.common.collect.ImmutableList; public class DefaultGameModeCommand extends VanillaCommand { + private static final List GAMEMODE_NAMES = ImmutableList.of("adventure", "creative", "survival"); + public DefaultGameModeCommand() { super("defaultgamemode"); this.description = "Set the default gamemode"; @@ -51,4 +60,17 @@ public class DefaultGameModeCommand extends VanillaCommand { return true; } + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + if (args.length == 1) { + return StringUtil.copyPartialMatches(args[0], GAMEMODE_NAMES, new ArrayList(GAMEMODE_NAMES.size())); + } else if (args.length == 2) { + return super.tabComplete(sender, alias, args); + } + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/DeopCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/DeopCommand.java index 291c049772..1ea6cadbf4 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/DeopCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/DeopCommand.java @@ -1,11 +1,17 @@ package org.bukkit.command.defaults; +import java.util.ArrayList; import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.ChatColor; import org.bukkit.OfflinePlayer; import org.bukkit.command.CommandSender; import org.bukkit.entity.Player; +import org.bukkit.util.StringUtil; + +import com.google.common.collect.ImmutableList; public class DeopCommand extends VanillaCommand { public DeopCommand() { @@ -34,13 +40,27 @@ public class DeopCommand extends VanillaCommand { return true; } - @Override - public List tabComplete(CommandSender sender, String[] args) { - return args.length >= 1 ? null : EMPTY_LIST; - } - @Override public boolean matches(String input) { return input.equalsIgnoreCase("deop"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + if (args.length == 1) { + List completions = new ArrayList(); + for (OfflinePlayer player : Bukkit.getOfflinePlayers()) { + String playerName = player.getName(); + if (player.isOp() && StringUtil.startsWithIgnoreCase(playerName, args[0])) { + completions.add(playerName); + } + } + return completions; + } + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/ExpCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/ExpCommand.java index 981cfb7b4e..791d2e1428 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/ExpCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/ExpCommand.java @@ -1,11 +1,16 @@ package org.bukkit.command.defaults; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.ChatColor; import org.bukkit.command.Command; import org.bukkit.command.CommandSender; import org.bukkit.entity.Player; +import com.google.common.collect.ImmutableList; + public class ExpCommand extends VanillaCommand { public ExpCommand() { super("xp"); @@ -44,4 +49,16 @@ public class ExpCommand extends VanillaCommand { public boolean matches(String input) { return input.equalsIgnoreCase("xp"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + if (args.length == 2) { + return super.tabComplete(sender, alias, args); + } + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/GameModeCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/GameModeCommand.java index ad44516030..8721e20b38 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/GameModeCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/GameModeCommand.java @@ -1,13 +1,22 @@ package org.bukkit.command.defaults; +import java.util.ArrayList; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.ChatColor; -import org.bukkit.command.CommandSender; -import org.bukkit.entity.Player; import org.bukkit.GameMode; import org.bukkit.command.Command; +import org.bukkit.command.CommandSender; +import org.bukkit.entity.Player; +import org.bukkit.util.StringUtil; + +import com.google.common.collect.ImmutableList; public class GameModeCommand extends VanillaCommand { + private static final List GAMEMODE_NAMES = ImmutableList.of("adventure", "creative", "survival"); + public GameModeCommand() { super("gamemode"); this.description = "Changes the player to a specific game mode"; @@ -77,4 +86,18 @@ public class GameModeCommand extends VanillaCommand { public boolean matches(String input) { return input.equalsIgnoreCase("gamemode"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + if (args.length == 1) { + return StringUtil.copyPartialMatches(args[0], GAMEMODE_NAMES, new ArrayList(GAMEMODE_NAMES.size())); + } else if (args.length == 2) { + return super.tabComplete(sender, alias, args); + } + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/GiveCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/GiveCommand.java index eeac739aad..6010ac86cc 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/GiveCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/GiveCommand.java @@ -1,5 +1,10 @@ package org.bukkit.command.defaults; +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.ChatColor; import org.bukkit.Material; @@ -7,8 +12,21 @@ import org.bukkit.command.Command; import org.bukkit.command.CommandSender; import org.bukkit.entity.Player; import org.bukkit.inventory.ItemStack; +import org.bukkit.util.StringUtil; + +import com.google.common.collect.ImmutableList; public class GiveCommand extends VanillaCommand { + private static List materials; + static { + ArrayList materialList = new ArrayList(); + for (Material material : Material.values()) { + materialList.add(material.name()); + } + Collections.sort(materialList); + materials = ImmutableList.copyOf(materialList); + } + public GiveCommand() { super("give"); this.description = "Gives the specified player a certain amount of items"; @@ -60,4 +78,45 @@ public class GiveCommand extends VanillaCommand { public boolean matches(String input) { return input.equalsIgnoreCase("give"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + if (args.length == 1) { + return super.tabComplete(sender, alias, args); + } + if (args.length == 2) { + final String arg = args[1]; + final List materials = GiveCommand.materials; + List completion = null; + + final int size = materials.size(); + int i = Collections.binarySearch(materials, arg, String.CASE_INSENSITIVE_ORDER); + + if (i < 0) { + // Insertion (start) index + i = -1 - i; + } + + for ( ; i < size; i++) { + String material = materials.get(i); + if (StringUtil.startsWithIgnoreCase(material, arg)) { + if (completion == null) { + completion = new ArrayList(); + } + completion.add(material); + } else { + break; + } + } + + if (completion != null) { + return completion; + } + } + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/HelpCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/HelpCommand.java index 4c002eb147..db3d92d01e 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/HelpCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/HelpCommand.java @@ -1,7 +1,15 @@ package org.bukkit.command.defaults; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.Set; +import java.util.TreeSet; + import org.apache.commons.lang.ArrayUtils; import org.apache.commons.lang.StringUtils; +import org.apache.commons.lang.Validate; import org.apache.commons.lang.math.NumberUtils; import org.bukkit.Bukkit; import org.bukkit.ChatColor; @@ -13,7 +21,7 @@ import org.bukkit.help.HelpTopicComparator; import org.bukkit.help.IndexHelpTopic; import org.bukkit.util.ChatPaginator; -import java.util.*; +import com.google.common.collect.ImmutableList; public class HelpCommand extends VanillaCommand { public HelpCommand() { @@ -106,6 +114,27 @@ public class HelpCommand extends VanillaCommand { return input.equalsIgnoreCase("help") || input.equalsIgnoreCase("?"); } + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + if (args.length == 1) { + List matchedTopics = new ArrayList(); + String searchString = args[0]; + for (HelpTopic topic : Bukkit.getServer().getHelpMap().getHelpTopics()) { + String trimmedTopic = topic.getName().startsWith("/") ? topic.getName().substring(1) : topic.getName(); + + if (trimmedTopic.startsWith(searchString)) { + matchedTopics.add(trimmedTopic); + } + } + return matchedTopics; + } + return ImmutableList.of(); + } + protected HelpTopic findPossibleMatches(String searchString) { int maxDistance = (searchString.length() / 5) + 3; Set possibleMatches = new TreeSet(HelpTopicComparator.helpTopicComparatorInstance()); diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/KickCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/KickCommand.java index 1f747d8c1b..7b890b1ae7 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/KickCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/KickCommand.java @@ -1,11 +1,16 @@ package org.bukkit.command.defaults; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.ChatColor; import org.bukkit.command.Command; import org.bukkit.command.CommandSender; import org.bukkit.entity.Player; +import com.google.common.collect.ImmutableList; + public class KickCommand extends VanillaCommand { public KickCommand() { super("kick"); @@ -44,4 +49,16 @@ public class KickCommand extends VanillaCommand { public boolean matches(String input) { return input.equalsIgnoreCase("kick"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + if (args.length >= 1) { + return super.tabComplete(sender, alias, args); + } + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/KillCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/KillCommand.java index d3049f7122..84f783f25e 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/KillCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/KillCommand.java @@ -1,10 +1,15 @@ package org.bukkit.command.defaults; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.command.CommandSender; import org.bukkit.entity.Player; import org.bukkit.event.entity.EntityDamageEvent; +import com.google.common.collect.ImmutableList; + public class KillCommand extends VanillaCommand { public KillCommand() { super("kill"); @@ -38,4 +43,13 @@ public class KillCommand extends VanillaCommand { public boolean matches(String input) { return input.equalsIgnoreCase("kill"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/ListCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/ListCommand.java index 489edd2889..e38ae80609 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/ListCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/ListCommand.java @@ -1,9 +1,14 @@ package org.bukkit.command.defaults; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.command.CommandSender; import org.bukkit.entity.Player; +import com.google.common.collect.ImmutableList; + public class ListCommand extends VanillaCommand { public ListCommand() { super("list"); @@ -41,4 +46,13 @@ public class ListCommand extends VanillaCommand { public boolean matches(String input) { return input.equalsIgnoreCase("list"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/OpCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/OpCommand.java index 4774c7e349..d7874bcf84 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/OpCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/OpCommand.java @@ -1,10 +1,19 @@ package org.bukkit.command.defaults; +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.ChatColor; import org.bukkit.OfflinePlayer; import org.bukkit.command.Command; import org.bukkit.command.CommandSender; +import org.bukkit.entity.Player; +import org.bukkit.util.StringUtil; + +import com.google.common.collect.ImmutableList; public class OpCommand extends VanillaCommand { public OpCommand() { @@ -33,4 +42,39 @@ public class OpCommand extends VanillaCommand { public boolean matches(String input) { return input.equalsIgnoreCase("op"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + if (args.length == 1) { + if (!(sender instanceof Player)) { + return ImmutableList.of(); + } + + String lastWord = args[0]; + if (lastWord.length() == 0) { + return ImmutableList.of(); + } + + Player senderPlayer = (Player) sender; + + ArrayList matchedPlayers = new ArrayList(); + for (Player player : sender.getServer().getOnlinePlayers()) { + String name = player.getName(); + if (!senderPlayer.canSee(player) || player.isOp()) { + continue; + } + if (StringUtil.startsWithIgnoreCase(name, lastWord)) { + matchedPlayers.add(name); + } + } + + Collections.sort(matchedPlayers, String.CASE_INSENSITIVE_ORDER); + return matchedPlayers; + } + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/PardonCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/PardonCommand.java index 29da200c63..a737ff0913 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/PardonCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/PardonCommand.java @@ -1,9 +1,17 @@ package org.bukkit.command.defaults; +import java.util.ArrayList; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.ChatColor; +import org.bukkit.OfflinePlayer; import org.bukkit.command.Command; import org.bukkit.command.CommandSender; +import org.bukkit.util.StringUtil; + +import com.google.common.collect.ImmutableList; public class PardonCommand extends VanillaCommand { public PardonCommand() { @@ -30,4 +38,23 @@ public class PardonCommand extends VanillaCommand { public boolean matches(String input) { return input.equalsIgnoreCase("pardon"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + if (args.length == 1) { + List completions = new ArrayList(); + for (OfflinePlayer player : Bukkit.getBannedPlayers()) { + String name = player.getName(); + if (StringUtil.startsWithIgnoreCase(name, args[0])) { + completions.add(name); + } + } + return completions; + } + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/PardonIpCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/PardonIpCommand.java index d97421100c..4571eeb99f 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/PardonIpCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/PardonIpCommand.java @@ -1,9 +1,16 @@ package org.bukkit.command.defaults; +import java.util.ArrayList; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.ChatColor; import org.bukkit.command.Command; import org.bukkit.command.CommandSender; +import org.bukkit.util.StringUtil; + +import com.google.common.collect.ImmutableList; public class PardonIpCommand extends VanillaCommand { public PardonIpCommand() { @@ -35,4 +42,16 @@ public class PardonIpCommand extends VanillaCommand { public boolean matches(String input) { return input.equalsIgnoreCase("pardon-ip"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + if (args.length == 1) { + return StringUtil.copyPartialMatches(args[0], Bukkit.getIPBans(), new ArrayList()); + } + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/PluginsCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/PluginsCommand.java index 8cb45cce23..b888da13ad 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/PluginsCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/PluginsCommand.java @@ -1,8 +1,9 @@ package org.bukkit.command.defaults; import java.util.Arrays; -import org.bukkit.ChatColor; + import org.bukkit.Bukkit; +import org.bukkit.ChatColor; import org.bukkit.command.CommandSender; import org.bukkit.plugin.Plugin; diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/ReloadCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/ReloadCommand.java index 9b090edcd0..7f820b3b3a 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/ReloadCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/ReloadCommand.java @@ -1,8 +1,9 @@ package org.bukkit.command.defaults; import java.util.Arrays; -import org.bukkit.ChatColor; + import org.bukkit.Bukkit; +import org.bukkit.ChatColor; import org.bukkit.command.CommandSender; public class ReloadCommand extends BukkitCommand { diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/SaveCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/SaveCommand.java index 08d8edf9a8..77839c1c7c 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/SaveCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/SaveCommand.java @@ -1,10 +1,15 @@ package org.bukkit.command.defaults; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.World; import org.bukkit.command.Command; import org.bukkit.command.CommandSender; +import com.google.common.collect.ImmutableList; + public class SaveCommand extends VanillaCommand { public SaveCommand() { super("save-all"); @@ -34,4 +39,13 @@ public class SaveCommand extends VanillaCommand { public boolean matches(String input) { return input.equalsIgnoreCase("save-all"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/SaveOffCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/SaveOffCommand.java index d46cdd3615..c081e5e106 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/SaveOffCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/SaveOffCommand.java @@ -1,10 +1,15 @@ package org.bukkit.command.defaults; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.World; import org.bukkit.command.Command; import org.bukkit.command.CommandSender; +import com.google.common.collect.ImmutableList; + public class SaveOffCommand extends VanillaCommand { public SaveOffCommand() { super("save-off"); @@ -29,4 +34,13 @@ public class SaveOffCommand extends VanillaCommand { public boolean matches(String input) { return input.equalsIgnoreCase("save-off"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/SaveOnCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/SaveOnCommand.java index 4c1bfec633..d0561c36a9 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/SaveOnCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/SaveOnCommand.java @@ -1,10 +1,15 @@ package org.bukkit.command.defaults; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.World; import org.bukkit.command.Command; import org.bukkit.command.CommandSender; +import com.google.common.collect.ImmutableList; + public class SaveOnCommand extends VanillaCommand { public SaveOnCommand() { super("save-on"); @@ -29,4 +34,13 @@ public class SaveOnCommand extends VanillaCommand { public boolean matches(String input) { return input.equalsIgnoreCase("save-on"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/SayCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/SayCommand.java index 44dc7ae0d9..651b60c0b1 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/SayCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/SayCommand.java @@ -1,10 +1,15 @@ package org.bukkit.command.defaults; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.ChatColor; import org.bukkit.command.CommandSender; import org.bukkit.entity.Player; +import com.google.common.collect.ImmutableList; + public class SayCommand extends VanillaCommand { public SayCommand() { super("say"); @@ -43,4 +48,15 @@ public class SayCommand extends VanillaCommand { public boolean matches(String input) { return input.equalsIgnoreCase("say"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + + if (args.length >= 1) { + return super.tabComplete(sender, alias, args); + } + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/SeedCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/SeedCommand.java index 4fac2678de..b64fd4051a 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/SeedCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/SeedCommand.java @@ -1,9 +1,14 @@ package org.bukkit.command.defaults; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.command.CommandSender; import org.bukkit.entity.Player; +import com.google.common.collect.ImmutableList; + public class SeedCommand extends VanillaCommand { public SeedCommand() { super("seed"); @@ -24,4 +29,13 @@ public class SeedCommand extends VanillaCommand { sender.sendMessage("Seed: " + seed); return true; } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/StopCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/StopCommand.java index 2b8ee08250..c6a470b646 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/StopCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/StopCommand.java @@ -1,9 +1,14 @@ package org.bukkit.command.defaults; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.command.Command; import org.bukkit.command.CommandSender; +import com.google.common.collect.ImmutableList; + public class StopCommand extends VanillaCommand { public StopCommand() { super("stop"); @@ -26,4 +31,13 @@ public class StopCommand extends VanillaCommand { public boolean matches(String input) { return input.equalsIgnoreCase("stop"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/TeleportCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/TeleportCommand.java index ad2a5527fc..55be9da355 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/TeleportCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/TeleportCommand.java @@ -1,5 +1,8 @@ package org.bukkit.command.defaults; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.ChatColor; import org.bukkit.Location; @@ -8,6 +11,8 @@ import org.bukkit.command.CommandSender; import org.bukkit.entity.Player; import org.bukkit.event.player.PlayerTeleportEvent.TeleportCause; +import com.google.common.collect.ImmutableList; + public class TeleportCommand extends VanillaCommand { public TeleportCommand() { super("tp"); @@ -66,4 +71,16 @@ public class TeleportCommand extends VanillaCommand { public boolean matches(String input) { return input.equalsIgnoreCase("tp"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + if (args.length == 1 || args.length == 2) { + return super.tabComplete(sender, alias, args); + } + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/TimeCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/TimeCommand.java index 9866877d25..011d32db2f 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/TimeCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/TimeCommand.java @@ -1,12 +1,22 @@ package org.bukkit.command.defaults; +import java.util.ArrayList; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.ChatColor; import org.bukkit.World; import org.bukkit.command.Command; import org.bukkit.command.CommandSender; +import org.bukkit.util.StringUtil; + +import com.google.common.collect.ImmutableList; public class TimeCommand extends VanillaCommand { + private static final List TABCOMPLETE_ADD_SET = ImmutableList.of("add", "set"); + private static final List TABCOMPLETE_DAY_NIGHT = ImmutableList.of("day", "night"); + public TimeCommand() { super("time"); this.description = "Changes the time on each world"; @@ -66,4 +76,18 @@ public class TimeCommand extends VanillaCommand { public boolean matches(String input) { return input.equalsIgnoreCase("time"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + if (args.length == 1) { + return StringUtil.copyPartialMatches(args[0], TABCOMPLETE_ADD_SET, new ArrayList(TABCOMPLETE_ADD_SET.size())); + } else if (args.length == 2 && args[0].equalsIgnoreCase("set")) { + return StringUtil.copyPartialMatches(args[1], TABCOMPLETE_DAY_NIGHT, new ArrayList(TABCOMPLETE_DAY_NIGHT.size())); + } + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/TimingsCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/TimingsCommand.java index b9c8fc6630..29ebbe050b 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/TimingsCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/TimingsCommand.java @@ -3,18 +3,25 @@ package org.bukkit.command.defaults; import java.io.File; import java.io.IOException; import java.io.PrintStream; +import java.util.ArrayList; +import java.util.List; +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.ChatColor; -import org.bukkit.command.Command; import org.bukkit.command.CommandSender; import org.bukkit.event.Event; import org.bukkit.event.HandlerList; import org.bukkit.plugin.Plugin; import org.bukkit.plugin.RegisteredListener; import org.bukkit.plugin.TimedRegisteredListener; +import org.bukkit.util.StringUtil; + +import com.google.common.collect.ImmutableList; public class TimingsCommand extends BukkitCommand { + private static final List TIMINGS_SUBCOMMANDS = ImmutableList.of("merged", "reset", "separate"); + public TimingsCommand(String name) { super(name); this.description = "Records timings for all plugin events"; @@ -99,4 +106,16 @@ public class TimingsCommand extends BukkitCommand { } return true; } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + if (args.length == 1) { + return StringUtil.copyPartialMatches(args[0], TIMINGS_SUBCOMMANDS, new ArrayList(TIMINGS_SUBCOMMANDS.size())); + } + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/ToggleDownfallCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/ToggleDownfallCommand.java index e81710e128..08ed667f4b 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/ToggleDownfallCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/ToggleDownfallCommand.java @@ -1,5 +1,8 @@ package org.bukkit.command.defaults; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.ChatColor; import org.bukkit.World; @@ -7,6 +10,8 @@ import org.bukkit.command.Command; import org.bukkit.command.CommandSender; import org.bukkit.entity.Player; +import com.google.common.collect.ImmutableList; + public class ToggleDownfallCommand extends VanillaCommand { public ToggleDownfallCommand() { super("toggledownfall"); @@ -44,4 +49,13 @@ public class ToggleDownfallCommand extends VanillaCommand { public boolean matches(String input) { return input.equalsIgnoreCase("toggledownfall"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) throws IllegalArgumentException { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/VanillaCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/VanillaCommand.java index bf6df75039..1fdcc81ee0 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/VanillaCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/VanillaCommand.java @@ -1,13 +1,11 @@ package org.bukkit.command.defaults; -import java.util.ArrayList; import java.util.List; + import org.bukkit.command.Command; import org.bukkit.command.CommandSender; public abstract class VanillaCommand extends Command { - static final List EMPTY_LIST = new ArrayList(0); - protected VanillaCommand(String name) { super(name); } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/VersionCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/VersionCommand.java index 4d3ba15b09..902f9d1585 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/VersionCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/VersionCommand.java @@ -1,13 +1,18 @@ package org.bukkit.command.defaults; +import java.util.ArrayList; import java.util.Arrays; import java.util.List; -import org.bukkit.ChatColor; +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; +import org.bukkit.ChatColor; import org.bukkit.command.CommandSender; import org.bukkit.plugin.Plugin; import org.bukkit.plugin.PluginDescriptionFile; +import org.bukkit.util.StringUtil; + +import com.google.common.collect.ImmutableList; public class VersionCommand extends BukkitCommand { public VersionCommand(String name) { @@ -102,4 +107,23 @@ public class VersionCommand extends BukkitCommand { return result.toString(); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + if (args.length == 1) { + List completions = new ArrayList(); + String toComplete = args[0].toLowerCase(); + for (Plugin plugin : Bukkit.getPluginManager().getPlugins()) { + if (StringUtil.startsWithIgnoreCase(plugin.getName(), toComplete)) { + completions.add(plugin.getName()); + } + } + return completions; + } + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/command/defaults/WhitelistCommand.java b/paper-api/src/main/java/org/bukkit/command/defaults/WhitelistCommand.java index d0fd85786f..70786d11e6 100644 --- a/paper-api/src/main/java/org/bukkit/command/defaults/WhitelistCommand.java +++ b/paper-api/src/main/java/org/bukkit/command/defaults/WhitelistCommand.java @@ -1,12 +1,21 @@ package org.bukkit.command.defaults; +import java.util.ArrayList; +import java.util.List; + +import org.apache.commons.lang.Validate; import org.bukkit.Bukkit; import org.bukkit.ChatColor; import org.bukkit.OfflinePlayer; import org.bukkit.command.Command; import org.bukkit.command.CommandSender; +import org.bukkit.util.StringUtil; + +import com.google.common.collect.ImmutableList; public class WhitelistCommand extends VanillaCommand { + private static final List WHITELIST_SUBCOMMANDS = ImmutableList.of("add", "remove", "on", "off", "list", "reload"); + public WhitelistCommand() { super("whitelist"); this.description = "Prevents the specified player from using this server"; @@ -88,4 +97,36 @@ public class WhitelistCommand extends VanillaCommand { public boolean matches(String input) { return input.equalsIgnoreCase("whitelist"); } + + @Override + public List tabComplete(CommandSender sender, String alias, String[] args) { + Validate.notNull(sender, "Sender cannot be null"); + Validate.notNull(args, "Arguments cannot be null"); + Validate.notNull(alias, "Alias cannot be null"); + + if (args.length == 1) { + return StringUtil.copyPartialMatches(args[0], WHITELIST_SUBCOMMANDS, new ArrayList(WHITELIST_SUBCOMMANDS.size())); + } else if (args.length == 2) { + if (args[0].equalsIgnoreCase("add")) { + List completions = new ArrayList(); + for (OfflinePlayer player : Bukkit.getOfflinePlayers()) { + String name = player.getName(); + if (StringUtil.startsWithIgnoreCase(name, args[1]) && !player.isWhitelisted()) { + completions.add(name); + } + } + return completions; + } else if (args[0].equalsIgnoreCase("remove")) { + List completions = new ArrayList(); + for (OfflinePlayer player : Bukkit.getWhitelistedPlayers()) { + String name = player.getName(); + if (StringUtil.startsWithIgnoreCase(name, args[1])) { + completions.add(name); + } + } + return completions; + } + } + return ImmutableList.of(); + } } diff --git a/paper-api/src/main/java/org/bukkit/plugin/Plugin.java b/paper-api/src/main/java/org/bukkit/plugin/Plugin.java index 6b75c3e7a6..c266114c5e 100644 --- a/paper-api/src/main/java/org/bukkit/plugin/Plugin.java +++ b/paper-api/src/main/java/org/bukkit/plugin/Plugin.java @@ -5,7 +5,7 @@ import java.io.InputStream; import java.util.logging.Logger; import org.bukkit.Server; -import org.bukkit.command.CommandExecutor; +import org.bukkit.command.TabExecutor; import org.bukkit.configuration.file.FileConfiguration; import org.bukkit.generator.ChunkGenerator; @@ -16,7 +16,7 @@ import com.avaje.ebean.EbeanServer; *

* The use of {@link PluginBase} is recommended for actual Implementation */ -public interface Plugin extends CommandExecutor { +public interface Plugin extends TabExecutor { /** * Returns the folder that the plugin data's files are located in. The * folder may not yet exist. diff --git a/paper-api/src/main/java/org/bukkit/plugin/java/JavaPlugin.java b/paper-api/src/main/java/org/bukkit/plugin/java/JavaPlugin.java index cdc6e75175..d48355476a 100644 --- a/paper-api/src/main/java/org/bukkit/plugin/java/JavaPlugin.java +++ b/paper-api/src/main/java/org/bukkit/plugin/java/JavaPlugin.java @@ -298,6 +298,13 @@ public abstract class JavaPlugin extends PluginBase { return false; } + /** + * {@inheritDoc} + */ + public List onTabComplete(CommandSender sender, Command command, String alias, String[] args) { + return null; + } + /** * Gets the command with the given name, specific to this plugin * diff --git a/paper-api/src/main/java/org/bukkit/util/StringUtil.java b/paper-api/src/main/java/org/bukkit/util/StringUtil.java new file mode 100644 index 0000000000..12bc58d4db --- /dev/null +++ b/paper-api/src/main/java/org/bukkit/util/StringUtil.java @@ -0,0 +1,49 @@ +package org.bukkit.util; + +import java.util.Collection; +import org.apache.commons.lang.Validate; + +public class StringUtil { + + /** + * Copies all elements from the iterable collection of originals to the collection provided. + * + * @param token String to search for + * @param originals An iterable collection of strings to filter. + * @param collection The collection to add matches to + * @return the collection provided that would have the elements copied into + * @throws UnsupportedOperationException if the collection is immutable and originals contains a string which starts with the specified search string. + * @throws IllegalArgumentException if any parameter is is null + * @throws IllegalArgumentException if originals contains a null element. Note: the collection may be modified before this is thrown + */ + public static > T copyPartialMatches(final String token, final Iterable originals, final T collection) throws UnsupportedOperationException, IllegalArgumentException { + Validate.notNull(token, "Search token cannot be null"); + Validate.notNull(collection, "Collection cannot be null"); + Validate.notNull(originals, "Originals cannot be null"); + + for (String string : originals) { + if (startsWithIgnoreCase(string, token)) { + collection.add(string); + } + } + + return collection; + } + + /** + * This method uses a substring to check case-insensitive equality. This means the internal array does not need to be copied like a toLowerCase() call would. + * + * @param string String to check + * @param prefix Prefix of string to compare + * @return true if provided string starts with, ignoring case, the prefix provided + * @throws NullPointerException if prefix is null + * @throws IllegalArgumentException if string is null + */ + public static boolean startsWithIgnoreCase(final String string, final String prefix) throws IllegalArgumentException, NullPointerException { + Validate.notNull(string, "Cannot check a null string for a match"); + if (string.length() < prefix.length()) { + return false; + } + return string.substring(0, prefix.length()).equalsIgnoreCase(prefix); + } +} diff --git a/paper-api/src/test/java/org/bukkit/plugin/TestPlugin.java b/paper-api/src/test/java/org/bukkit/plugin/TestPlugin.java index a435771aed..7e098925cd 100644 --- a/paper-api/src/test/java/org/bukkit/plugin/TestPlugin.java +++ b/paper-api/src/test/java/org/bukkit/plugin/TestPlugin.java @@ -2,6 +2,7 @@ package org.bukkit.plugin; import java.io.File; import java.io.InputStream; +import java.util.List; import org.bukkit.Server; import org.bukkit.command.Command; @@ -103,4 +104,8 @@ public class TestPlugin extends PluginBase { public boolean onCommand(CommandSender sender, Command command, String label, String[] args) { throw new UnsupportedOperationException("Not supported."); } + + public List onTabComplete(CommandSender sender, Command command, String alias, String[] args) { + throw new UnsupportedOperationException("Not supported."); + } }