Add a JDA 5 module

Eventually will drop the JDA 4 module - but it still functions so no
reason to arbitrarily upgrade it or drop without bumping acf version
This commit is contained in:
chickeneer
2024-12-13 18:11:04 -06:00
parent 43323e656a
commit 623048b763
23 changed files with 875 additions and 2 deletions
+2 -2
View File
@@ -8,6 +8,7 @@
<sourceTestOutputDir name="target/generated-test-sources/test-annotations" />
<outputRelativeToContentRoot value="true" />
<module name="acf-jda" />
<module name="acf-jda5" />
<module name="acf-paper" />
<module name="example-plugin" />
<module name="acf" />
@@ -24,8 +25,6 @@
<bytecodeTargetLevel>
<module name="acf" target="1.8" />
<module name="acf-example" target="1.8" />
<module name="acf-sponge" target="1.8" />
<module name="acf-sponge10" target="1.8" />
<module name="commands" target="1.8" />
<module name="example-plugin" target="1.8" />
</bytecodeTargetLevel>
@@ -37,6 +36,7 @@
<module name="acf-bungee" options="-parameters" />
<module name="acf-core" options="-parameters" />
<module name="acf-jda" options="-parameters" />
<module name="acf-jda5" options="-parameters" />
<module name="acf-paper" options="-parameters" />
<module name="acf-parent" options="-parameters" />
<module name="acf-sponge" options="-parameters" />
+2
View File
@@ -17,6 +17,8 @@
<file url="file://$PROJECT_DIR$/jda" charset="UTF-8" />
<file url="file://$PROJECT_DIR$/jda/src/main/java" charset="UTF-8" />
<file url="file://$PROJECT_DIR$/jda/src/main/resources" charset="UTF-8" />
<file url="file://$PROJECT_DIR$/jda5/src/main/java" charset="UTF-8" />
<file url="file://$PROJECT_DIR$/jda5/src/main/resources" charset="UTF-8" />
<file url="file://$PROJECT_DIR$/languages/core" charset="UTF-8" />
<file url="file://$PROJECT_DIR$/languages/minecraft" charset="UTF-8" />
<file url="file://$PROJECT_DIR$/paper" charset="UTF-8" />
+63
View File
@@ -0,0 +1,63 @@
<?xml version="1.0" encoding="UTF-8"?>
<!--
~ Copyright (c) 2016-2017 Daniel Ennis (Aikar) - MIT License
~
~ 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.
-->
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>co.aikar</groupId>
<artifactId>acf-parent</artifactId>
<version><!--VERSION-->0.5.1-SNAPSHOT<!--VERSION--></version>
<relativePath>../pom.xml</relativePath>
</parent>
<artifactId>acf-jda5</artifactId>
<version><!--VERSION-->0.5.1-SNAPSHOT<!--VERSION--></version>
<name>ACF (JDA 5)</name>
<dependencies>
<dependency>
<groupId>co.aikar</groupId>
<artifactId>acf-core</artifactId>
<version><!--VERSION-->0.5.1-SNAPSHOT<!--VERSION--></version>
<scope>compile</scope>
</dependency>
<dependency>
<groupId>net.dv8tion</groupId>
<artifactId>JDA</artifactId>
<version>5.2.1</version>
<scope>provided</scope>
</dependency>
</dependencies>
<build>
<resources>
<!--resource>
<directory>${project.basedir}/../languages/minecraft/</directory>
</resource-->
</resources>
</build>
</project>
@@ -0,0 +1,15 @@
package co.aikar.commands;
import net.dv8tion.jda.api.events.message.MessageReceivedEvent;
import org.jetbrains.annotations.NotNull;
import java.util.List;
public interface CommandConfig extends CommandConfigProvider {
@NotNull List<String> getCommandPrefixes();
@Override
default CommandConfig provide(MessageReceivedEvent event) {
return this;
}
}
@@ -0,0 +1,7 @@
package co.aikar.commands;
import net.dv8tion.jda.api.events.message.MessageReceivedEvent;
public interface CommandConfigProvider {
CommandConfig provide(MessageReceivedEvent event);
}
@@ -0,0 +1,6 @@
package co.aikar.commands;
public interface CommandPermissionResolver {
boolean hasPermission(JDACommandManager manager, JDACommandEvent event, String permission);
}
@@ -0,0 +1,42 @@
package co.aikar.commands;
import org.jetbrains.annotations.NotNull;
import java.util.Collections;
import java.util.List;
public class JDACommandCompletions extends CommandCompletions<CommandCompletionContext<?>> {
private boolean initialized;
public JDACommandCompletions(CommandManager manager) {
super(manager);
this.initialized = true;
}
@Override
public CommandCompletionHandler registerCompletion(String id, CommandCompletionHandler<CommandCompletionContext<?>> handler) {
if (initialized) {
throw new UnsupportedOperationException("JDA Doesn't support Command Completions");
}
return null;
}
@Override
public CommandCompletionHandler registerAsyncCompletion(String id, AsyncCommandCompletionHandler<CommandCompletionContext<?>> handler) {
if (initialized) {
throw new UnsupportedOperationException("JDA Doesn't support Command Completions");
}
return null;
}
@NotNull
@Override
List<String> of(RegisteredCommand command, CommandIssuer sender, String[] args, boolean isAsync) {
return Collections.emptyList();
}
@Override
List<String> getCompletionValues(RegisteredCommand command, CommandIssuer sender, String completion, String[] args, boolean isAsync) {
return Collections.emptyList();
}
}
@@ -0,0 +1,19 @@
package co.aikar.commands;
import org.jetbrains.annotations.NotNull;
import java.util.List;
import java.util.concurrent.CopyOnWriteArrayList;
public class JDACommandConfig implements CommandConfig {
protected @NotNull List<String> commandPrefixes = new CopyOnWriteArrayList<>(new String[]{"!"});
public JDACommandConfig() {
}
@NotNull
public List<String> getCommandPrefixes() {
return commandPrefixes;
}
}
@@ -0,0 +1,115 @@
package co.aikar.commands;
import co.aikar.commands.annotation.Author;
import co.aikar.commands.annotation.CrossGuild;
import co.aikar.commands.annotation.SelfUser;
import net.dv8tion.jda.api.JDA;
import net.dv8tion.jda.api.entities.channel.ChannelType;
import net.dv8tion.jda.api.entities.Guild;
import net.dv8tion.jda.api.entities.Message;
import net.dv8tion.jda.api.entities.channel.middleman.MessageChannel;
import net.dv8tion.jda.api.entities.Role;
import net.dv8tion.jda.api.entities.channel.concrete.TextChannel;
import net.dv8tion.jda.api.entities.User;
import net.dv8tion.jda.api.events.message.MessageReceivedEvent;
import java.util.List;
// TODO: Message Keys !!!
public class JDACommandContexts extends CommandContexts<JDACommandExecutionContext> {
private final JDACommandManager manager;
private final JDA jda;
public JDACommandContexts(JDACommandManager manager) {
super(manager);
this.manager = manager;
this.jda = this.manager.getJDA();
this.registerIssuerOnlyContext(JDACommandEvent.class, CommandExecutionContext::getIssuer);
this.registerIssuerOnlyContext(MessageReceivedEvent.class, c -> c.getIssuer().getIssuer());
this.registerIssuerOnlyContext(Message.class, c -> c.issuer.getIssuer().getMessage());
this.registerIssuerOnlyContext(ChannelType.class, c -> c.issuer.getIssuer().getChannelType());
this.registerIssuerOnlyContext(JDA.class, c -> jda);
this.registerIssuerOnlyContext(Guild.class, c -> {
MessageReceivedEvent event = c.getIssuer().getIssuer();
if (event.isFromType(ChannelType.PRIVATE) && !c.isOptional()) {
throw new InvalidCommandArgument("This command can only be executed in a Guild.", false);
} else {
return event.getGuild();
}
});
this.registerIssuerAwareContext(MessageChannel.class, c -> {
if (c.hasAnnotation(Author.class)) {
return c.issuer.getIssuer().getChannel();
}
boolean isCrossGuild = c.hasAnnotation(CrossGuild.class);
String argument = c.popFirstArg(); // we pop because we are only issuer aware if we are annotated
MessageChannel channel = null;
if (argument.startsWith("<#")) {
String id = argument.substring(2, argument.length() - 1);
channel = isCrossGuild ? jda.getTextChannelById(id) : c.issuer.getIssuer().getGuild().getTextChannelById(id);
} else {
List<TextChannel> channelList = isCrossGuild ? jda.getTextChannelsByName(argument, true) :
c.issuer.getEvent().getGuild().getTextChannelsByName(argument, true);
if (channelList.size() > 1) {
throw new InvalidCommandArgument("Too many channels were found with the given name. Try with the `#channelname` syntax.", false);
} else if (channelList.size() == 1) {
channel = channelList.get(0);
}
}
if (channel == null) {
throw new InvalidCommandArgument("Couldn't find a channel with that name or ID.");
}
return channel;
});
this.registerIssuerAwareContext(User.class, c -> {
if (c.hasAnnotation(SelfUser.class)) {
return jda.getSelfUser();
}
String arg = c.getFirstArg();
if (c.isOptional() && (arg == null || arg.isEmpty())) {
return null;
}
arg = c.popFirstArg(); // we pop because we are only issuer aware if we are annotated
User user = null;
if (arg.startsWith("<@!")) { // for some reason a ! is added when @'ing and clicking their name.
user = jda.getUserById(arg.substring(3, arg.length() - 1));
} else if (arg.startsWith("<@")) { // users can /also/ be mentioned like this...
user = jda.getUserById(arg.substring(2, arg.length() - 1));
} else {
List<User> users = jda.getUsersByName(arg, true);
if (users.size() > 1) {
throw new InvalidCommandArgument("Too many users were found with the given name. Try with the `@username#0000` syntax.", false);
}
if (!users.isEmpty()) {
user = users.get(0);
}
}
if (user == null) {
throw new InvalidCommandArgument("Could not find a user with that name or ID.");
}
return user;
});
this.registerContext(Role.class, c -> {
boolean isCrossGuild = c.hasAnnotation(CrossGuild.class);
String arg = c.popFirstArg();
Role role = null;
if (arg.startsWith("<@&")) {
String id = arg.substring(3, arg.length() - 1);
role = isCrossGuild ? jda.getRoleById(id) : c.issuer.getIssuer().getGuild().getRoleById(id);
} else {
List<Role> roles = isCrossGuild ? jda.getRolesByName(arg, true)
: c.issuer.getIssuer().getGuild().getRolesByName(arg, true);
if (roles.size() > 1) {
throw new InvalidCommandArgument("Too many roles were found with the given name. Try with the `@role` syntax.", false);
}
if (!roles.isEmpty()) {
role = roles.get(0);
}
}
if (role == null) {
throw new InvalidCommandArgument("Could not find a role with that name or ID.");
}
return role;
});
}
}
@@ -0,0 +1,67 @@
package co.aikar.commands;
import net.dv8tion.jda.api.entities.Message;
import net.dv8tion.jda.api.entities.MessageEmbed;
import net.dv8tion.jda.api.events.message.MessageReceivedEvent;
import net.dv8tion.jda.api.utils.messages.MessageCreateData;
import org.jetbrains.annotations.NotNull;
import java.util.UUID;
public class JDACommandEvent implements CommandIssuer {
private MessageReceivedEvent event;
private JDACommandManager manager;
public JDACommandEvent(JDACommandManager manager, MessageReceivedEvent event) {
this.manager = manager;
this.event = event;
}
public MessageReceivedEvent getEvent() {
return event;
}
@Override
public MessageReceivedEvent getIssuer() {
return event;
}
@Override
public CommandManager getManager() {
return this.manager;
}
@Override
public boolean isPlayer() {
return false;
}
@Override
public @NotNull UUID getUniqueId() {
// Discord id only have 64 bit width (long) while UUIDs have twice the size.
// In order to keep it unique we use 0L for the first 64 bit.
long authorId = event.getAuthor().getIdLong();
return new UUID(0, authorId);
}
@Override
public boolean hasPermission(String permission) {
CommandPermissionResolver permissionResolver = this.manager.getPermissionResolver();
return permissionResolver == null || permissionResolver.hasPermission(manager, this, permission);
}
@Override
public void sendMessageInternal(String message) {
this.event.getChannel().sendMessage(message).queue();
}
public void sendMessage(Message message) {
this.event.getChannel().sendMessage(MessageCreateData.fromMessage(message)).queue();
}
public void sendMessage(MessageEmbed message) {
this.event.getChannel().sendMessage(MessageCreateData.fromEmbeds(message)).queue();
}
}
@@ -0,0 +1,10 @@
package co.aikar.commands;
import java.util.List;
import java.util.Map;
public class JDACommandExecutionContext extends CommandExecutionContext<JDACommandExecutionContext, JDACommandEvent> {
JDACommandExecutionContext(RegisteredCommand cmd, CommandParameter param, JDACommandEvent sender, List<String> args, int index, Map<String, Object> passedArgs) {
super(cmd, param, sender, args, index, passedArgs);
}
}
@@ -0,0 +1,285 @@
package co.aikar.commands;
import co.aikar.commands.apachecommonslang.ApacheCommonsExceptionUtil;
import net.dv8tion.jda.api.JDA;
import net.dv8tion.jda.api.entities.channel.ChannelType;
import net.dv8tion.jda.api.entities.Message;
import net.dv8tion.jda.api.events.message.MessageReceivedEvent;
import org.jetbrains.annotations.NotNull;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.logging.Level;
import java.util.logging.Logger;
public class JDACommandManager extends CommandManager<
MessageReceivedEvent,
JDACommandEvent,
String,
MessageFormatter<String>,
JDACommandExecutionContext,
JDAConditionContext
> {
private final JDA jda;
protected JDACommandCompletions completions;
protected JDACommandContexts contexts;
protected JDALocales locales;
protected Map<String, JDARootCommand> commands = new HashMap<>();
private Logger logger;
private CommandConfig defaultConfig;
private CommandConfigProvider configProvider;
private CommandPermissionResolver permissionResolver;
private long botOwner = 0L;
public JDACommandManager(JDA jda) {
this(jda, null);
}
public JDACommandManager(JDA jda, JDAOptions options) {
if (options == null) {
options = new JDAOptions();
}
this.jda = jda;
this.permissionResolver = options.permissionResolver;
jda.addEventListener(new JDAListener(this));
this.defaultConfig = options.defaultConfig == null ? new JDACommandConfig() : options.defaultConfig;
this.configProvider = options.configProvider;
this.defaultFormatter = new JDAMessageFormatter();
this.completions = new JDACommandCompletions(this);
this.logger = Logger.getLogger(this.getClass().getSimpleName());
getCommandConditions().addCondition("owneronly", context -> {
if (context.getIssuer().getEvent().getAuthor().getIdLong() != getBotOwnerId()) {
throw new ConditionFailedException("Only the bot owner can use this command."); // TODO: MessageKey
}
});
getCommandConditions().addCondition("guildonly", context -> {
if (context.getIssuer().getEvent().getChannelType() != ChannelType.TEXT) {
throw new ConditionFailedException("This command must be used in guild chat."); // TODO: MessageKey
}
});
getCommandConditions().addCondition("privateonly", context -> {
if (context.getIssuer().getEvent().getChannelType() != ChannelType.PRIVATE) {
throw new ConditionFailedException("This command must be used in private chat."); // TODO: MessageKey
}
});
getCommandConditions().addCondition("grouponly", context -> {
if (context.getIssuer().getEvent().getChannelType() != ChannelType.GROUP) {
throw new ConditionFailedException("This command must be used in group chat."); // TODO: MessageKey
}
});
}
public static JDAOptions options() {
return new JDAOptions();
}
void initializeBotOwner() {
if (botOwner == 0L) {
botOwner = jda.retrieveApplicationInfo().complete().getOwner().getIdLong();
}
}
public long getBotOwnerId() {
// Just in case initialization on ReadyEvent fails.
initializeBotOwner();
return botOwner;
}
public JDA getJDA() {
return jda;
}
public Logger getLogger() {
return logger;
}
public void setLogger(Logger logger) {
this.logger = logger;
}
public CommandConfig getDefaultConfig() {
return defaultConfig;
}
public void setDefaultConfig(@NotNull CommandConfig defaultConfig) {
this.defaultConfig = defaultConfig;
}
public CommandConfigProvider getConfigProvider() {
return configProvider;
}
public void setConfigProvider(CommandConfigProvider configProvider) {
this.configProvider = configProvider;
}
public CommandPermissionResolver getPermissionResolver() {
return permissionResolver;
}
public void setPermissionResolver(CommandPermissionResolver permissionResolver) {
this.permissionResolver = permissionResolver;
}
@Override
public CommandContexts<?> getCommandContexts() {
if (this.contexts == null) {
this.contexts = new JDACommandContexts(this);
}
return this.contexts;
}
@Override
public CommandCompletions<?> getCommandCompletions() {
return this.completions;
}
@Override
public void registerCommand(BaseCommand command) {
command.onRegister(this);
for (Map.Entry<String, RootCommand> entry : command.registeredCommands.entrySet()) {
String commandName = entry.getKey().toLowerCase(Locale.ENGLISH);
JDARootCommand cmd = (JDARootCommand) entry.getValue();
if (!cmd.isRegistered) {
cmd.isRegistered = true;
commands.put(commandName, cmd);
}
}
}
public void unregisterCommand(BaseCommand command) {
for (Map.Entry<String, RootCommand> entry : command.registeredCommands.entrySet()) {
String jdaCommandName = entry.getKey().toLowerCase(Locale.ENGLISH);
JDARootCommand jdaCommand = (JDARootCommand) entry.getValue();
jdaCommand.getSubCommands().values().removeAll(command.subCommands.values());
if (jdaCommand.isRegistered && jdaCommand.getSubCommands().isEmpty()) {
jdaCommand.isRegistered = false;
commands.remove(jdaCommandName);
}
}
}
@Override
public boolean hasRegisteredCommands() {
return !this.commands.isEmpty();
}
@Override
public boolean isCommandIssuer(Class<?> type) {
return JDACommandEvent.class.isAssignableFrom(type);
}
@Override
public JDACommandEvent getCommandIssuer(Object issuer) {
if (!(issuer instanceof MessageReceivedEvent)) {
throw new IllegalArgumentException(issuer.getClass().getName() + " is not a Message Received Event.");
}
return new JDACommandEvent(this, (MessageReceivedEvent) issuer);
}
@Override
public RootCommand createRootCommand(String cmd) {
return new JDARootCommand(this, cmd);
}
@Override
public Collection<RootCommand> getRegisteredRootCommands() {
return Collections.unmodifiableCollection(commands.values());
}
@Override
public Locales getLocales() {
if (this.locales == null) {
this.locales = new JDALocales(this);
this.locales.loadLanguages();
}
return this.locales;
}
@Override
public CommandExecutionContext createCommandContext(RegisteredCommand command, CommandParameter parameter, CommandIssuer sender, List<String> args, int i, Map<String, Object> passedArgs) {
return new JDACommandExecutionContext(command, parameter, (JDACommandEvent) sender, args, i, passedArgs);
}
@Override
public CommandCompletionContext createCompletionContext(RegisteredCommand command, CommandIssuer sender, String input, String config, String[] args) {
// Not really going to be used;
//noinspection unchecked
return new CommandCompletionContext(command, sender, input, config, args);
}
@Override
public void log(LogLevel level, String message, Throwable throwable) {
Level logLevel = level == LogLevel.INFO ? Level.INFO : Level.SEVERE;
logger.log(logLevel, LogLevel.LOG_PREFIX + message);
if (throwable != null) {
for (String line : ACFPatterns.NEWLINE.split(ApacheCommonsExceptionUtil.getFullStackTrace(throwable))) {
logger.log(logLevel, LogLevel.LOG_PREFIX + line);
}
}
}
void dispatchEvent(MessageReceivedEvent event) {
Message message = event.getMessage();
String msg = message.getContentRaw();
CommandConfig config = getCommandConfig(event);
String prefixFound = null;
for (String prefix : config.getCommandPrefixes()) {
if (msg.startsWith(prefix)) {
prefixFound = prefix;
break;
}
}
if (prefixFound == null) {
return;
}
String[] args = ACFPatterns.SPACE.split(msg.substring(prefixFound.length()), -1);
if (args.length == 0) {
return;
}
String cmd = args[0].toLowerCase(Locale.ENGLISH);
JDARootCommand rootCommand = this.commands.get(cmd);
if (rootCommand == null) {
return;
}
if (args.length > 1) {
args = Arrays.copyOfRange(args, 1, args.length);
} else {
args = new String[0];
}
rootCommand.execute(this.getCommandIssuer(event), cmd, args);
}
private CommandConfig getCommandConfig(MessageReceivedEvent event) {
CommandConfig config = this.defaultConfig;
if (this.configProvider != null) {
CommandConfig provided = this.configProvider.provide(event);
if (provided != null) {
config = provided;
}
}
return config;
}
@Override
public String getCommandPrefix(CommandIssuer issuer) {
MessageReceivedEvent event = ((JDACommandEvent) issuer).getEvent();
CommandConfig commandConfig = getCommandConfig(event);
List<String> prefixes = commandConfig.getCommandPrefixes();
return prefixes.isEmpty() ? "" : prefixes.get(0);
}
}
@@ -0,0 +1,40 @@
package co.aikar.commands;
import net.dv8tion.jda.api.Permission;
import java.util.HashMap;
import java.util.Locale;
import java.util.Map;
public class JDACommandPermissionResolver implements CommandPermissionResolver {
private Map<String, Integer> discordPermissionOffsets;
public JDACommandPermissionResolver() {
discordPermissionOffsets = new HashMap<>();
for (Permission permission : Permission.values()) {
discordPermissionOffsets.put(permission.name().toLowerCase(Locale.ENGLISH).replaceAll("_", "-"), permission.getOffset());
}
}
@Override
public boolean hasPermission(JDACommandManager manager, JDACommandEvent event, String permission) {
// Explicitly return true if the issuer is the bot's owner. They are always allowed.
if (manager.getBotOwnerId() == event.getIssuer().getAuthor().getIdLong()) {
return true;
}
// Return false on webhook messages, as they cannot have permissions defined.
if (event.getIssuer().isWebhookMessage()) {
return false;
}
Integer permissionOffset = discordPermissionOffsets.get(permission);
if (permissionOffset == null) {
return false;
}
return event.getIssuer().getMember().hasPermission(
Permission.getFromOffset(permissionOffset)
);
}
}
@@ -0,0 +1,7 @@
package co.aikar.commands;
public class JDAConditionContext extends ConditionContext<JDACommandEvent> {
JDAConditionContext(JDACommandEvent issuer, String config) {
super(issuer, config);
}
}
@@ -0,0 +1,33 @@
package co.aikar.commands;
import net.dv8tion.jda.api.entities.channel.ChannelType;
import net.dv8tion.jda.api.events.message.MessageReceivedEvent;
import net.dv8tion.jda.api.events.session.GenericSessionEvent;
import net.dv8tion.jda.api.events.session.SessionState;
import net.dv8tion.jda.api.hooks.ListenerAdapter;
import org.jetbrains.annotations.NotNull;
public class JDAListener extends ListenerAdapter {
private final JDACommandManager manager;
JDAListener(JDACommandManager manager) {
this.manager = manager;
}
@Override
public void onMessageReceived(MessageReceivedEvent event) {
if (event.isFromType(ChannelType.TEXT) || event.isFromType(ChannelType.PRIVATE)) {
this.manager.dispatchEvent(event);
}
}
@Override
public void onGenericSession(@NotNull GenericSessionEvent event) {
if (event.getState() == SessionState.READY) {
manager.initializeBotOwner();
}
}
}
@@ -0,0 +1,7 @@
package co.aikar.commands;
public class JDALocales extends Locales {
public JDALocales(CommandManager manager) {
super(manager);
}
}
@@ -0,0 +1,14 @@
package co.aikar.commands;
public class JDAMessageFormatter extends MessageFormatter<String> {
public JDAMessageFormatter() {
// JDA does not support coloring messages outside of embed fields.
// We pass three empty strings to remove color coded messages from appearing.
super("", "", "");
}
@Override
String format(String color, String message) {
return message;
}
}
@@ -0,0 +1,32 @@
package co.aikar.commands;
import net.dv8tion.jda.api.JDA;
import org.jetbrains.annotations.NotNull;
public class JDAOptions {
CommandConfig defaultConfig = new JDACommandConfig();
CommandConfigProvider configProvider = null;
CommandPermissionResolver permissionResolver = new JDACommandPermissionResolver();
public JDAOptions() {
}
public JDAOptions defaultConfig(@NotNull CommandConfig defaultConfig) {
this.defaultConfig = defaultConfig;
return this;
}
public JDAOptions configProvider(@NotNull CommandConfigProvider configProvider) {
this.configProvider = configProvider;
return this;
}
public JDAOptions permissionResolver(@NotNull CommandPermissionResolver permissionResolver) {
this.permissionResolver = permissionResolver;
return this;
}
public JDACommandManager create(JDA jda) {
return new JDACommandManager(jda, this);
}
}
@@ -0,0 +1,56 @@
package co.aikar.commands;
import com.google.common.collect.HashMultimap;
import com.google.common.collect.SetMultimap;
import java.util.ArrayList;
import java.util.List;
public class JDARootCommand implements RootCommand {
private final String name;
boolean isRegistered = false;
private JDACommandManager manager;
private BaseCommand defCommand;
private SetMultimap<String, RegisteredCommand> subCommands = HashMultimap.create();
private List<BaseCommand> children = new ArrayList<>();
JDARootCommand(JDACommandManager manager, String name) {
this.manager = manager;
this.name = name;
}
@Override
public void addChild(BaseCommand command) {
if (this.defCommand == null || !command.subCommands.get(BaseCommand.DEFAULT).isEmpty()) {
this.defCommand = command;
}
addChildShared(this.children, this.subCommands, command);
}
@Override
public CommandManager getManager() {
return this.manager;
}
@Override
public SetMultimap<String, RegisteredCommand> getSubCommands() {
return this.subCommands;
}
@Override
public List<BaseCommand> getChildren() {
return this.children;
}
@Override
public String getCommandName() {
return this.name;
}
@Override
public BaseCommand getDefCommand() {
return defCommand;
}
}
@@ -0,0 +1,18 @@
package co.aikar.commands.annotation;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/**
* The {@link Author} annotation is to define whether the parameter should be the author object from the event or
* parsed from the user's input.
* <p>
* Using this on a User/Member will fetch the author and otherwise it'll parse the input.
* </p>
*/
@Target(ElementType.PARAMETER)
@Retention(RetentionPolicy.RUNTIME)
public @interface Author {
}
@@ -0,0 +1,19 @@
package co.aikar.commands.annotation;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/**
* The {@link CrossGuild} annotation is to define whether the parameter should be guild-specific or global.
* <p>
* If a supported parameter is marked with the CrossGuild annotation, the parameter will be filled from
* a global perspective (i.e., all of the guilds the bot is connected to). Otherwise, the parameter will
* be filled from command input.
* </p>
*/
@Target(ElementType.PARAMETER)
@Retention(RetentionPolicy.RUNTIME)
public @interface CrossGuild {
}
@@ -0,0 +1,15 @@
package co.aikar.commands.annotation;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/**
* The {@link SelfUser} annotation is to define whether the parameter should be represented by JDA's user object
* or if it should be parsed from command input.
*/
@Target(ElementType.PARAMETER)
@Retention(RetentionPolicy.RUNTIME)
public @interface SelfUser {
}
+1
View File
@@ -206,6 +206,7 @@
<module>sponge</module>
<module>sponge10</module>
<module>jda</module>
<module>jda5</module>
<module>velocity</module>
</modules>
</project>