Move to class-based command handling
All checks were successful
continuous-integration/drone/push Build is passing
All checks were successful
continuous-integration/drone/push Build is passing
Having everything in a single class is bad practice, so different classes for each command were made.
This commit is contained in:
parent
dd4ffe252e
commit
a2c1944a32
@ -12,6 +12,10 @@ import net.dv8tion.jda.api.requests.restaction.interactions.ReplyCallbackAction;
|
|||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
import wtf.beatrice.hidekobot.Configuration;
|
import wtf.beatrice.hidekobot.Configuration;
|
||||||
import wtf.beatrice.hidekobot.HidekoBot;
|
import wtf.beatrice.hidekobot.HidekoBot;
|
||||||
|
import wtf.beatrice.hidekobot.slashcommands.ClearChatCommand;
|
||||||
|
import wtf.beatrice.hidekobot.slashcommands.CoinFlipCommand;
|
||||||
|
import wtf.beatrice.hidekobot.slashcommands.DieCommand;
|
||||||
|
import wtf.beatrice.hidekobot.slashcommands.PingCommand;
|
||||||
import wtf.beatrice.hidekobot.utils.RandomUtil;
|
import wtf.beatrice.hidekobot.utils.RandomUtil;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
@ -25,170 +29,11 @@ public class SlashCommandListener extends ListenerAdapter
|
|||||||
@Override
|
@Override
|
||||||
public void onSlashCommandInteraction(@NotNull SlashCommandInteractionEvent event)
|
public void onSlashCommandInteraction(@NotNull SlashCommandInteractionEvent event)
|
||||||
{
|
{
|
||||||
if (event.getName().equals("ping"))
|
switch (event.getName().toLowerCase()) {
|
||||||
{
|
case "ping" -> new PingCommand(event);
|
||||||
event.reply("Pong!").queue();
|
case "die" -> new DieCommand(event);
|
||||||
}
|
case "coinflip" -> new CoinFlipCommand(event);
|
||||||
|
case "clear" -> new ClearChatCommand(event);
|
||||||
else if (event.getName().equals("die"))
|
|
||||||
{
|
|
||||||
if(Configuration.getBotOwnerId() != event.getMember().getIdLong())
|
|
||||||
{
|
|
||||||
event.reply("Sorry, only the bot owner can run this command!").setEphemeral(true).queue();
|
|
||||||
} else {
|
|
||||||
event.reply("Going to sleep! Cya :sparkles:").queue();
|
|
||||||
Executors.newSingleThreadScheduledExecutor().schedule(HidekoBot::shutdown, 3, TimeUnit.SECONDS);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
else if (event.getName().equals("coinflip"))
|
|
||||||
{
|
|
||||||
int rand = RandomUtil.getRandomNumber(0, 1);
|
|
||||||
String msg;
|
|
||||||
|
|
||||||
if(rand == 1)
|
|
||||||
{
|
|
||||||
msg = ":coin: It's **Heads**!";
|
|
||||||
} else {
|
|
||||||
msg = "It's **Tails**! :coin:";
|
|
||||||
}
|
|
||||||
|
|
||||||
event.reply(msg).queue();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
else if (event.getName().equals("clear"))
|
|
||||||
{
|
|
||||||
MessageChannel channel = event.getChannel();
|
|
||||||
|
|
||||||
if(!(channel instanceof TextChannel))
|
|
||||||
{
|
|
||||||
event.reply("Sorry! I can't delete messages here.").queue();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* get the amount from the command args.
|
|
||||||
NULL should not be possible because we specified them as mandatory,
|
|
||||||
but apparently the mobile app doesn't care and still sends the command if you omit the args. */
|
|
||||||
OptionMapping amountMapping = event.getOption("amount");
|
|
||||||
int toDeleteAmount = amountMapping == null ? 1 : amountMapping.getAsInt();
|
|
||||||
|
|
||||||
if(toDeleteAmount <= 0)
|
|
||||||
{
|
|
||||||
event.reply("Sorry, I can't delete that amount of messages!").queue();
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
// answer by saying that the operation has begun.
|
|
||||||
InteractionHook replyInteraction = event.reply("\uD83D\uDEA7 Clearing...").complete();
|
|
||||||
|
|
||||||
// int to keep track of how many messages we deleted.
|
|
||||||
int deleted = 0;
|
|
||||||
|
|
||||||
int limit = 95; //discord limits this method to range 2-100. we set it to 95 to be safe..
|
|
||||||
|
|
||||||
// increase the count by 1, because we technically aren't clearing the first ID ever
|
|
||||||
// which is actually the slash command's ID and not a message.
|
|
||||||
toDeleteAmount++;
|
|
||||||
|
|
||||||
// count how many times we have to iterate this to delete the full <toDeleteAmount> messages.
|
|
||||||
int iterations = toDeleteAmount / limit;
|
|
||||||
|
|
||||||
//if there are some messages left, but less than <limit>, we need one more iterations.
|
|
||||||
int remainder = toDeleteAmount % limit;
|
|
||||||
if(remainder != 0) iterations++;
|
|
||||||
|
|
||||||
// set the starting point.
|
|
||||||
long messageId = event.getInteraction().getIdLong();
|
|
||||||
|
|
||||||
// boolean to see if we're trying to delete more messages than possible.
|
|
||||||
boolean outOfBounds = false;
|
|
||||||
|
|
||||||
// do iterate.
|
|
||||||
for(int iteration = 0; iteration < iterations; iteration++)
|
|
||||||
{
|
|
||||||
if(outOfBounds) break;
|
|
||||||
|
|
||||||
// set how many messages to delete for this iteration (usually <limit> unless there's a remainder)
|
|
||||||
int iterationSize = limit;
|
|
||||||
|
|
||||||
// if we are at the last iteration...
|
|
||||||
if(iteration+1 == iterations)
|
|
||||||
{
|
|
||||||
// check if we have <limit> or fewer messages to delete
|
|
||||||
if(remainder != 0) iterationSize = remainder;
|
|
||||||
}
|
|
||||||
|
|
||||||
if(iterationSize == 1)
|
|
||||||
{
|
|
||||||
// grab the message
|
|
||||||
Message toDelete = ((TextChannel)channel).retrieveMessageById(messageId).complete();
|
|
||||||
//only delete one message
|
|
||||||
if(toDelete != null) toDelete.delete().queue();
|
|
||||||
else outOfBounds = true;
|
|
||||||
// increase deleted counter by 1
|
|
||||||
deleted++;
|
|
||||||
} else {
|
|
||||||
// get the last <iterationSize - 1> messages.
|
|
||||||
MessageHistory.MessageRetrieveAction action = channel.getHistoryBefore(messageId, iterationSize - 1);
|
|
||||||
// note: first one is the most recent, last one is the oldest message.
|
|
||||||
List<Message> messages = new ArrayList<>();
|
|
||||||
// (we are skipping first iteration since it would return an error, given that the id is the slash command and not a message)
|
|
||||||
if(iteration!=0) messages.add(((TextChannel)channel).retrieveMessageById(messageId).complete());
|
|
||||||
messages.addAll(action.complete().getRetrievedHistory());
|
|
||||||
|
|
||||||
// check if we only have one or zero messages left (trying to delete more than possible)
|
|
||||||
if(messages.size() <= 1)
|
|
||||||
{
|
|
||||||
outOfBounds = true;
|
|
||||||
} else {
|
|
||||||
// before deleting, we need to grab the <previous to the oldest> message's id for next iteration.
|
|
||||||
action = channel.getHistoryBefore(messages.get(messages.size() - 1).getIdLong(), 1);
|
|
||||||
|
|
||||||
List<Message> previousMessage = action.complete().getRetrievedHistory();
|
|
||||||
|
|
||||||
// if that message exists (we are not out of bounds)... store it
|
|
||||||
if(!previousMessage.isEmpty()) messageId = previousMessage.get(0).getIdLong();
|
|
||||||
else outOfBounds = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// queue messages for deletion
|
|
||||||
if(messages.size() == 1)
|
|
||||||
{
|
|
||||||
messages.get(0).delete().queue();
|
|
||||||
}
|
|
||||||
else if(!messages.isEmpty())
|
|
||||||
{
|
|
||||||
try {
|
|
||||||
((TextChannel) channel).deleteMessages(messages).complete();
|
|
||||||
/* alternatively, we could use purgeMessages, which is smarter...
|
|
||||||
however, it also tries to delete messages older than 2 weeks
|
|
||||||
which are restricted by discord, and thus has to use
|
|
||||||
a less efficient way that triggers rate-limiting very quickly. */
|
|
||||||
} catch (Exception e)
|
|
||||||
{
|
|
||||||
replyInteraction.editOriginal("\uD83D\uDE22 Sorry, it seems like there was an issue! " + e.getMessage()).queue();
|
|
||||||
return; // warning: this quits everything.
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// increase deleted counter by <list size>
|
|
||||||
deleted += messages.size();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// log having deleted the messages.
|
|
||||||
if(deleted < 1)
|
|
||||||
{
|
|
||||||
replyInteraction.editOriginal("\uD83D\uDE22 Couldn't clear any message!").queue();
|
|
||||||
} else if(deleted == 1)
|
|
||||||
{
|
|
||||||
replyInteraction.editOriginal("✂ Cleared 1 message!").queue();
|
|
||||||
} else {
|
|
||||||
replyInteraction.editOriginal("✂ Cleared " + deleted + " messages!").queue();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,150 @@
|
|||||||
|
package wtf.beatrice.hidekobot.slashcommands;
|
||||||
|
|
||||||
|
import net.dv8tion.jda.api.entities.Message;
|
||||||
|
import net.dv8tion.jda.api.entities.MessageChannel;
|
||||||
|
import net.dv8tion.jda.api.entities.MessageHistory;
|
||||||
|
import net.dv8tion.jda.api.entities.TextChannel;
|
||||||
|
import net.dv8tion.jda.api.events.interaction.command.SlashCommandInteractionEvent;
|
||||||
|
import net.dv8tion.jda.api.interactions.InteractionHook;
|
||||||
|
import net.dv8tion.jda.api.interactions.commands.OptionMapping;
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
public class ClearChatCommand
|
||||||
|
{
|
||||||
|
|
||||||
|
public ClearChatCommand(@NotNull SlashCommandInteractionEvent event)
|
||||||
|
{
|
||||||
|
MessageChannel channel = event.getChannel();
|
||||||
|
|
||||||
|
if(!(channel instanceof TextChannel))
|
||||||
|
{
|
||||||
|
event.reply("Sorry! I can't delete messages here.").queue();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* get the amount from the command args.
|
||||||
|
NULL should not be possible because we specified them as mandatory,
|
||||||
|
but apparently the mobile app doesn't care and still sends the command if you omit the args. */
|
||||||
|
OptionMapping amountMapping = event.getOption("amount");
|
||||||
|
int toDeleteAmount = amountMapping == null ? 1 : amountMapping.getAsInt();
|
||||||
|
|
||||||
|
if(toDeleteAmount <= 0)
|
||||||
|
{
|
||||||
|
event.reply("Sorry, I can't delete that amount of messages!").queue();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// answer by saying that the operation has begun.
|
||||||
|
InteractionHook replyInteraction = event.reply("\uD83D\uDEA7 Clearing...").complete();
|
||||||
|
|
||||||
|
// int to keep track of how many messages we deleted.
|
||||||
|
int deleted = 0;
|
||||||
|
|
||||||
|
int limit = 95; //discord limits this method to range 2-100. we set it to 95 to be safe..
|
||||||
|
|
||||||
|
// increase the count by 1, because we technically aren't clearing the first ID ever
|
||||||
|
// which is actually the slash command's ID and not a message.
|
||||||
|
toDeleteAmount++;
|
||||||
|
|
||||||
|
// count how many times we have to iterate this to delete the full <toDeleteAmount> messages.
|
||||||
|
int iterations = toDeleteAmount / limit;
|
||||||
|
|
||||||
|
//if there are some messages left, but less than <limit>, we need one more iterations.
|
||||||
|
int remainder = toDeleteAmount % limit;
|
||||||
|
if(remainder != 0) iterations++;
|
||||||
|
|
||||||
|
// set the starting point.
|
||||||
|
long messageId = event.getInteraction().getIdLong();
|
||||||
|
|
||||||
|
// boolean to see if we're trying to delete more messages than possible.
|
||||||
|
boolean outOfBounds = false;
|
||||||
|
|
||||||
|
// do iterate.
|
||||||
|
for(int iteration = 0; iteration < iterations; iteration++)
|
||||||
|
{
|
||||||
|
if(outOfBounds) break;
|
||||||
|
|
||||||
|
// set how many messages to delete for this iteration (usually <limit> unless there's a remainder)
|
||||||
|
int iterationSize = limit;
|
||||||
|
|
||||||
|
// if we are at the last iteration...
|
||||||
|
if(iteration+1 == iterations)
|
||||||
|
{
|
||||||
|
// check if we have <limit> or fewer messages to delete
|
||||||
|
if(remainder != 0) iterationSize = remainder;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(iterationSize == 1)
|
||||||
|
{
|
||||||
|
// grab the message
|
||||||
|
Message toDelete = ((TextChannel)channel).retrieveMessageById(messageId).complete();
|
||||||
|
//only delete one message
|
||||||
|
if(toDelete != null) toDelete.delete().queue();
|
||||||
|
else outOfBounds = true;
|
||||||
|
// increase deleted counter by 1
|
||||||
|
deleted++;
|
||||||
|
} else {
|
||||||
|
// get the last <iterationSize - 1> messages.
|
||||||
|
MessageHistory.MessageRetrieveAction action = channel.getHistoryBefore(messageId, iterationSize - 1);
|
||||||
|
// note: first one is the most recent, last one is the oldest message.
|
||||||
|
List<Message> messages = new ArrayList<>();
|
||||||
|
// (we are skipping first iteration since it would return an error, given that the id is the slash command and not a message)
|
||||||
|
if(iteration!=0) messages.add(((TextChannel)channel).retrieveMessageById(messageId).complete());
|
||||||
|
messages.addAll(action.complete().getRetrievedHistory());
|
||||||
|
|
||||||
|
// check if we only have one or zero messages left (trying to delete more than possible)
|
||||||
|
if(messages.size() <= 1)
|
||||||
|
{
|
||||||
|
outOfBounds = true;
|
||||||
|
} else {
|
||||||
|
// before deleting, we need to grab the <previous to the oldest> message's id for next iteration.
|
||||||
|
action = channel.getHistoryBefore(messages.get(messages.size() - 1).getIdLong(), 1);
|
||||||
|
|
||||||
|
List<Message> previousMessage = action.complete().getRetrievedHistory();
|
||||||
|
|
||||||
|
// if that message exists (we are not out of bounds)... store it
|
||||||
|
if(!previousMessage.isEmpty()) messageId = previousMessage.get(0).getIdLong();
|
||||||
|
else outOfBounds = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// queue messages for deletion
|
||||||
|
if(messages.size() == 1)
|
||||||
|
{
|
||||||
|
messages.get(0).delete().queue();
|
||||||
|
}
|
||||||
|
else if(!messages.isEmpty())
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
((TextChannel) channel).deleteMessages(messages).complete();
|
||||||
|
/* alternatively, we could use purgeMessages, which is smarter...
|
||||||
|
however, it also tries to delete messages older than 2 weeks
|
||||||
|
which are restricted by discord, and thus has to use
|
||||||
|
a less efficient way that triggers rate-limiting very quickly. */
|
||||||
|
} catch (Exception e)
|
||||||
|
{
|
||||||
|
replyInteraction.editOriginal("\uD83D\uDE22 Sorry, it seems like there was an issue! " + e.getMessage()).queue();
|
||||||
|
return; // warning: this quits everything.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// increase deleted counter by <list size>
|
||||||
|
deleted += messages.size();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// log having deleted the messages.
|
||||||
|
if(deleted < 1)
|
||||||
|
{
|
||||||
|
replyInteraction.editOriginal("\uD83D\uDE22 Couldn't clear any message!").queue();
|
||||||
|
} else if(deleted == 1)
|
||||||
|
{
|
||||||
|
replyInteraction.editOriginal("✂ Cleared 1 message!").queue();
|
||||||
|
} else {
|
||||||
|
replyInteraction.editOriginal("✂ Cleared " + deleted + " messages!").queue();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,24 @@
|
|||||||
|
package wtf.beatrice.hidekobot.slashcommands;
|
||||||
|
|
||||||
|
import net.dv8tion.jda.api.events.interaction.command.SlashCommandInteractionEvent;
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
import wtf.beatrice.hidekobot.utils.RandomUtil;
|
||||||
|
|
||||||
|
public class CoinFlipCommand
|
||||||
|
{
|
||||||
|
|
||||||
|
public CoinFlipCommand(@NotNull SlashCommandInteractionEvent event)
|
||||||
|
{
|
||||||
|
int rand = RandomUtil.getRandomNumber(0, 1);
|
||||||
|
String msg;
|
||||||
|
|
||||||
|
if(rand == 1)
|
||||||
|
{
|
||||||
|
msg = ":coin: It's **Heads**!";
|
||||||
|
} else {
|
||||||
|
msg = "It's **Tails**! :coin:";
|
||||||
|
}
|
||||||
|
|
||||||
|
event.reply(msg).queue();
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,24 @@
|
|||||||
|
package wtf.beatrice.hidekobot.slashcommands;
|
||||||
|
|
||||||
|
import net.dv8tion.jda.api.events.interaction.command.SlashCommandInteractionEvent;
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
import wtf.beatrice.hidekobot.Configuration;
|
||||||
|
import wtf.beatrice.hidekobot.HidekoBot;
|
||||||
|
|
||||||
|
import java.util.concurrent.Executors;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
public class DieCommand
|
||||||
|
{
|
||||||
|
|
||||||
|
public DieCommand(@NotNull SlashCommandInteractionEvent event)
|
||||||
|
{
|
||||||
|
if(Configuration.getBotOwnerId() != event.getMember().getIdLong())
|
||||||
|
{
|
||||||
|
event.reply("Sorry, only the bot owner can run this command!").setEphemeral(true).queue();
|
||||||
|
} else {
|
||||||
|
event.reply("Going to sleep! Cya :sparkles:").queue();
|
||||||
|
Executors.newSingleThreadScheduledExecutor().schedule(HidekoBot::shutdown, 3, TimeUnit.SECONDS);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,12 @@
|
|||||||
|
package wtf.beatrice.hidekobot.slashcommands;
|
||||||
|
|
||||||
|
import net.dv8tion.jda.api.events.interaction.command.SlashCommandInteractionEvent;
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
|
||||||
|
public class PingCommand
|
||||||
|
{
|
||||||
|
public PingCommand(@NotNull SlashCommandInteractionEvent event)
|
||||||
|
{
|
||||||
|
event.reply("Pong!").queue();
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user