summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorBuildTools <OfficialNemes@gmail.com>2017-07-22 12:31:16 +0200
committerBuildTools <OfficialNemes@gmail.com>2017-07-22 12:31:16 +0200
commit508aceb7176b082f58d3eeb324cf618346d99fc0 (patch)
treed670ed0d411617afa5cec0ff67cea71637fd0ef8
Initial commit
-rw-r--r--net/nemez/chatapi/ChatAPI.java132
-rw-r--r--net/nemez/chatapi/click/CallbackCommand.java38
-rw-r--r--net/nemez/chatapi/click/CallbackMap.java53
-rw-r--r--net/nemez/chatapi/click/ClickCallback.java40
-rw-r--r--net/nemez/chatapi/click/Message.java131
-rw-r--r--net/nemez/chatapi/click/PlayerQuitListener.java13
-rw-r--r--net/nemez/chatapi/click/RunnableCallback.java17
7 files changed, 424 insertions, 0 deletions
diff --git a/net/nemez/chatapi/ChatAPI.java b/net/nemez/chatapi/ChatAPI.java
new file mode 100644
index 0000000..a5af0d0
--- /dev/null
+++ b/net/nemez/chatapi/ChatAPI.java
@@ -0,0 +1,132 @@
+package net.nemez.chatapi;
+
+import java.lang.reflect.Field;
+import java.util.Random;
+
+import org.bukkit.Bukkit;
+import org.bukkit.command.CommandMap;
+import org.bukkit.command.CommandSender;
+import org.bukkit.plugin.java.JavaPlugin;
+
+import net.nemez.chatapi.click.CallbackCommand;
+import net.nemez.chatapi.click.Message;
+import net.nemez.chatapi.click.PlayerQuitListener;
+
+public class ChatAPI {
+
+ /* message coloring permission */
+ public static final String PERMISSION_CHAT_COLOR = "chat.color";
+ /* message formatting permission */
+ public static final String PERMISSION_CHAT_FORMAT = "chat.format";
+ /* message magic formatting permission */
+ public static final String PERMISSION_CHAT_MAGIC = "chat.magic";
+ /* permission to send messages in chat */
+ public static final String PERMISSION_CHAT_USE = "chat.use";
+ /* message to send when the internal command is not ran correctly (ran by user) */
+ public static final String MESSAGE_HELP_CLICK_CALLBACK = "&cThis is an internal command for ChatAPI and should not be ran by players manually.";
+ /* message to send when the internal command is not ran by a player */
+ public static final String MESSAGE_PLAYER_CLICK_CALLBACK = "&cThis command can only be run by a player";
+ /* the actual command name for use in click callbacks */
+ private static String internalCommandName;
+
+ /**
+ * Initializes ChatAPI and registers the required commands for clickable chat to function.
+ */
+ public static void initialize(JavaPlugin plugin) {
+ if (internalCommandName != null) {
+ return;
+ }
+ Random rand = new Random(System.currentTimeMillis());
+ internalCommandName = "chatapi-exec-" + Integer.toHexString(rand.nextInt(0xEFFF) + 0x1000);
+ try {
+ final Field cmdMap = Bukkit.getServer().getClass().getDeclaredField("commandMap");
+ cmdMap.setAccessible(true);
+ CommandMap map = (CommandMap) cmdMap.get(Bukkit.getServer());
+ map.register("chatapi", new CallbackCommand(internalCommandName));
+ } catch (Exception e) {
+ plugin.getLogger().severe("Failed to register internal command '" + internalCommandName + "'");
+ e.printStackTrace();
+ }
+ internalCommandName = "chatapi:" + internalCommandName;
+ plugin.getServer().getPluginManager().registerEvents(new PlayerQuitListener(), plugin);
+ plugin.getLogger().info("ChatAPI initialized");
+ }
+
+ /**
+ * Colorifies a message using &format codes. Respects permissions.
+ *
+ * @param sender the command sender whose permissions to use. null if permissions are to be ignored.
+ * @param message the message to color
+ * @return colored message
+ */
+ public static String colorify(CommandSender sender, String message) {
+ if (sender == null || sender.hasPermission(PERMISSION_CHAT_COLOR)) {
+ message = message.replaceAll("&([0-9a-fA-FrR])", "§$1");
+ }
+ if (sender == null || sender.hasPermission(PERMISSION_CHAT_FORMAT)) {
+ message = message.replaceAll("&([l-oL-OrR])", "§$1");
+ }
+ if (sender == null || sender.hasPermission(PERMISSION_CHAT_MAGIC)) {
+ message = message.replaceAll("&([kKrR])", "§$1");
+ }
+ return message.replaceAll("&§", "&");
+ }
+
+ /**
+ * Sends a colorified message to the command sender.
+ *
+ * @param sender the command sender to whom to send the message.
+ * @param message the message to send.
+ */
+ public static void send(CommandSender sender, String message) {
+ if (sender == null) {
+ return;
+ }
+ sender.sendMessage(colorify(null, message));
+ }
+
+ /**
+ * Checks if a command sender has the permission node required to send chat messages.
+ *
+ * @param sender the command sender to check.
+ * @return true/false if sender can chat or is null.
+ */
+ public static boolean canChat(CommandSender sender) {
+ if (sender == null) {
+ return true;
+ }else{
+ return sender.hasPermission(PERMISSION_CHAT_USE);
+ }
+ }
+
+ /**
+ * Creates a new message object that will be sent to the given command sender with regards to the second command sender's permissions.
+ *
+ * @param sender the command sender to whom to send the message.
+ * @param permissionSender the command sender whose permissions to use.
+ * @return message object
+ */
+ public static Message createMessage(CommandSender sender, CommandSender permissionSender) {
+ return new Message(sender, permissionSender);
+ }
+
+ /**
+ * Creates a new message object that will be sent to the given command sender.
+ *
+ * @param sender the command sender to whom to send the message.
+ * @return message object.
+ */
+ public static Message createMessage(CommandSender sender) {
+ return createMessage(sender, null);
+ }
+
+ /**
+ * Gets the name of the internal ChatAPI command used for click callbacks.
+ * This function is used internally and you don't need to worry about it.
+ *
+ * @return callback command name
+ */
+ public static String getInternalCallbackCommand() {
+ return internalCommandName;
+ }
+}
diff --git a/net/nemez/chatapi/click/CallbackCommand.java b/net/nemez/chatapi/click/CallbackCommand.java
new file mode 100644
index 0000000..ead17bd
--- /dev/null
+++ b/net/nemez/chatapi/click/CallbackCommand.java
@@ -0,0 +1,38 @@
+package net.nemez.chatapi.click;
+
+import java.util.UUID;
+
+import org.bukkit.command.Command;
+import org.bukkit.command.CommandSender;
+import org.bukkit.entity.Player;
+
+import net.nemez.chatapi.ChatAPI;
+
+public class CallbackCommand extends Command {
+
+ public CallbackCommand(String internalCommandName) {
+ super(internalCommandName);
+ }
+
+ @Override
+ public boolean execute(CommandSender sender, String label, String[] args) {
+ if (!(sender instanceof Player)) {
+ ChatAPI.send(sender, ChatAPI.MESSAGE_PLAYER_CLICK_CALLBACK);
+ return true;
+ }
+ if ((args.length == 1 && args[0].equals("help")) || args.length != 1) {
+ ChatAPI.send(sender, ChatAPI.MESSAGE_HELP_CLICK_CALLBACK);
+ return true;
+ }
+ int id;
+ try {
+ id = Integer.parseInt(args[0]);
+ } catch (NumberFormatException e) {
+ ChatAPI.send(sender, ChatAPI.MESSAGE_HELP_CLICK_CALLBACK);
+ return true;
+ }
+ UUID uuid = ((Player) sender).getUniqueId();
+ CallbackMap.execute(sender, uuid, id);
+ return true;
+ }
+}
diff --git a/net/nemez/chatapi/click/CallbackMap.java b/net/nemez/chatapi/click/CallbackMap.java
new file mode 100644
index 0000000..3a8f996
--- /dev/null
+++ b/net/nemez/chatapi/click/CallbackMap.java
@@ -0,0 +1,53 @@
+package net.nemez.chatapi.click;
+
+import java.util.HashMap;
+import java.util.UUID;
+
+import org.bukkit.command.CommandSender;
+
+public class CallbackMap {
+
+ private static HashMap<String, HashMap<Integer, ClickCallback>> map = new HashMap<String, HashMap<Integer, ClickCallback>>();
+
+ protected static int register(UUID uuid, ClickCallback callback) {
+ HashMap<Integer, ClickCallback> playerMap = map.get(uuid.toString());
+ if (playerMap == null) {
+ playerMap = new HashMap<Integer, ClickCallback>();
+ map.put(uuid.toString(), playerMap);
+ }
+ int largestId = 0;
+ for (int i : playerMap.keySet()) {
+ if (i > largestId) {
+ largestId = i;
+ }
+ }
+ int id = largestId + 1;
+ playerMap.put(id, callback);
+ return id;
+ }
+
+ protected static void execute(CommandSender sender, UUID uuid, int id) {
+ HashMap<Integer, ClickCallback> playerMap = map.get(uuid.toString());
+ if (playerMap == null) {
+ return;
+ }
+ ClickCallback cb = playerMap.get(id);
+ if (cb == null) {
+ return;
+ }
+ if (cb.isAsynchronous()) {
+ Thread t = new Thread() {
+ public void run() {
+ cb.execute(sender);
+ }
+ };
+ t.start();
+ }else{
+ cb.execute(sender);
+ }
+ }
+
+ protected static void discard(UUID uuid) {
+ map.remove(uuid.toString());
+ }
+}
diff --git a/net/nemez/chatapi/click/ClickCallback.java b/net/nemez/chatapi/click/ClickCallback.java
new file mode 100644
index 0000000..c2d2a87
--- /dev/null
+++ b/net/nemez/chatapi/click/ClickCallback.java
@@ -0,0 +1,40 @@
+package net.nemez.chatapi.click;
+
+import org.bukkit.command.CommandSender;
+
+public abstract class ClickCallback {
+
+ private boolean repeatable, async;
+ private String expiredMessage;
+ private boolean expired;
+
+ public ClickCallback(boolean repeatable, boolean async, String expiredMessage) {
+ this.repeatable = repeatable;
+ this.async = async;
+ this.expiredMessage = expiredMessage;
+ this.expired = false;
+ }
+
+ public abstract void run(CommandSender sender);
+
+ public final void execute(CommandSender sender) {
+ if (!expired) {
+ run(sender);
+ }
+ if (!repeatable) {
+ expired = true;
+ }
+ }
+
+ public boolean isRepeatable() {
+ return repeatable;
+ }
+
+ public boolean isAsynchronous() {
+ return async;
+ }
+
+ public String getExpiredMessage() {
+ return expiredMessage;
+ }
+}
diff --git a/net/nemez/chatapi/click/Message.java b/net/nemez/chatapi/click/Message.java
new file mode 100644
index 0000000..0abb578
--- /dev/null
+++ b/net/nemez/chatapi/click/Message.java
@@ -0,0 +1,131 @@
+package net.nemez.chatapi.click;
+
+import org.bukkit.command.CommandSender;
+import org.bukkit.entity.Player;
+
+import net.md_5.bungee.api.chat.ClickEvent;
+import net.md_5.bungee.api.chat.ComponentBuilder;
+import net.md_5.bungee.api.chat.HoverEvent;
+import net.md_5.bungee.api.chat.TextComponent;
+import net.nemez.chatapi.ChatAPI;
+
+public class Message {
+
+ private CommandSender sender;
+ private CommandSender permission;
+ private TextComponent message;
+ private String rawMessage;
+
+ public Message(CommandSender sender, CommandSender permission) {
+ this.sender = sender;
+ this.permission = permission;
+ message = new TextComponent("");
+ rawMessage = "";
+ }
+
+ public Message appendText(String text) {
+ text = ChatAPI.colorify(permission, text);
+ message.addExtra(text);
+ rawMessage += text;
+ return this;
+ }
+
+ public Message appendLink(String text, String url) {
+ text = ChatAPI.colorify(permission, text);
+ TextComponent component = new TextComponent(text);
+ component.setClickEvent(new ClickEvent(ClickEvent.Action.OPEN_URL, url));
+ message.addExtra(component);
+ rawMessage += text;
+ return this;
+ }
+
+ public Message appendSendChat(String text, String msg) {
+ text = ChatAPI.colorify(permission, text);
+ TextComponent component = new TextComponent(text);
+ component.setClickEvent(new ClickEvent(ClickEvent.Action.RUN_COMMAND, msg));
+ message.addExtra(component);
+ rawMessage += text;
+ return this;
+ }
+
+ public Message appendSuggest(String text, String suggestion) {
+ text = ChatAPI.colorify(permission, text);
+ TextComponent component = new TextComponent(text);
+ component.setClickEvent(new ClickEvent(ClickEvent.Action.SUGGEST_COMMAND, suggestion));
+ message.addExtra(component);
+ rawMessage += text;
+ return this;
+ }
+
+ public Message appendCallback(String text, ClickCallback callback) {
+ if (sender instanceof Player) {
+ int id = CallbackMap.register(((Player) sender).getUniqueId(), callback);
+ return appendSendChat(text, "/" + ChatAPI.getInternalCallbackCommand() + " " + id);
+ }else{
+ return appendText(text);
+ }
+ }
+
+ public Message appendTextHover(String text, String hover) {
+ text = ChatAPI.colorify(permission, text);
+ TextComponent component = new TextComponent(text);
+ addHoverText(component, hover);
+ message.addExtra(component);
+ rawMessage += text;
+ return this;
+ }
+
+ public Message appendLinkHover(String text, String url, String hover) {
+ text = ChatAPI.colorify(permission, text);
+ TextComponent component = new TextComponent(text);
+ component.setClickEvent(new ClickEvent(ClickEvent.Action.OPEN_URL, url));
+ addHoverText(component, hover);
+ message.addExtra(component);
+ rawMessage += text;
+ return this;
+ }
+
+ public Message appendSendChatHover(String text, String msg, String hover) {
+ text = ChatAPI.colorify(permission, text);
+ TextComponent component = new TextComponent(text);
+ component.setClickEvent(new ClickEvent(ClickEvent.Action.RUN_COMMAND, msg));
+ addHoverText(component, hover);
+ message.addExtra(component);
+ rawMessage += text;
+ return this;
+ }
+
+ public Message appendSuggestHover(String text, String suggestion, String hover) {
+ text = ChatAPI.colorify(permission, text);
+ TextComponent component = new TextComponent(text);
+ component.setClickEvent(new ClickEvent(ClickEvent.Action.SUGGEST_COMMAND, suggestion));
+ addHoverText(component, hover);
+ message.addExtra(component);
+ rawMessage += text;
+ return this;
+ }
+
+ public Message appendCallbackHover(String text, ClickCallback callback, String hover) {
+ if (sender instanceof Player) {
+ int id = CallbackMap.register(((Player) sender).getUniqueId(), callback);
+ return appendSendChatHover(text, "/" + ChatAPI.getInternalCallbackCommand() + " " + id, hover);
+ }else{
+ return appendTextHover(text, hover);
+ }
+ }
+
+ public void send() {
+ if (sender == null || !ChatAPI.canChat(this.permission)) {
+ return;
+ }
+ if (sender instanceof Player) {
+ ((Player)sender).spigot().sendMessage(message);
+ }else{
+ sender.sendMessage(rawMessage);
+ }
+ }
+
+ private void addHoverText(TextComponent comp, String text) {
+ comp.setHoverEvent(new HoverEvent(HoverEvent.Action.SHOW_TEXT, new ComponentBuilder(ChatAPI.colorify(permission, text)).create()));
+ }
+}
diff --git a/net/nemez/chatapi/click/PlayerQuitListener.java b/net/nemez/chatapi/click/PlayerQuitListener.java
new file mode 100644
index 0000000..04e7c1b
--- /dev/null
+++ b/net/nemez/chatapi/click/PlayerQuitListener.java
@@ -0,0 +1,13 @@
+package net.nemez.chatapi.click;
+
+import org.bukkit.event.EventHandler;
+import org.bukkit.event.Listener;
+import org.bukkit.event.player.PlayerQuitEvent;
+
+public class PlayerQuitListener implements Listener {
+
+ @EventHandler
+ public void onPlayerQuit(PlayerQuitEvent event) {
+ CallbackMap.discard(event.getPlayer().getUniqueId());
+ }
+}
diff --git a/net/nemez/chatapi/click/RunnableCallback.java b/net/nemez/chatapi/click/RunnableCallback.java
new file mode 100644
index 0000000..0b4f42f
--- /dev/null
+++ b/net/nemez/chatapi/click/RunnableCallback.java
@@ -0,0 +1,17 @@
+package net.nemez.chatapi.click;
+
+import org.bukkit.command.CommandSender;
+
+public class RunnableCallback extends ClickCallback {
+
+ private Runnable runnable;
+
+ public RunnableCallback(Runnable runnable, boolean repeatable, boolean async, String expiredMessage) {
+ super(repeatable, async, expiredMessage);
+ }
+
+ @Override
+ public void run(CommandSender sender) {
+ runnable.run();
+ }
+}