MiaoChat_For_Paper1.18.1/src/main/java/pw/yumc/MiaoChat/listeners/ChatListener.java

204 lines
7.4 KiB
Java

package pw.yumc.MiaoChat.listeners;
import me.clip.placeholderapi.PlaceholderAPI;
import org.bukkit.Bukkit;
import org.bukkit.ChatColor;
import org.bukkit.Material;
import org.bukkit.entity.Entity;
import org.bukkit.entity.Player;
import org.bukkit.event.EventHandler;
import org.bukkit.event.EventPriority;
import org.bukkit.event.Listener;
import org.bukkit.event.player.AsyncPlayerChatEvent;
import org.bukkit.inventory.ItemStack;
import pw.yumc.MiaoChat.MiaoChat;
import pw.yumc.MiaoChat.MiaoMessage;
import pw.yumc.MiaoChat.config.ChatConfig;
import pw.yumc.MiaoChat.config.ChatMessagePart;
import pw.yumc.MiaoChat.config.ChatRule;
import pw.yumc.YumCore.bukkit.Log;
import pw.yumc.YumCore.bukkit.P;
import pw.yumc.YumCore.bukkit.compatible.C;
import pw.yumc.YumCore.global.L10N;
import pw.yumc.YumCore.statistic.Statistics;
import pw.yumc.YumCore.tellraw.Tellraw;
import pw.yumc.YumCore.update.SubscribeTask;
import java.util.*;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
public class ChatListener implements Listener {
public static Set<String> offList = new HashSet<>();
private static Pattern ITEM_PATTERN = Pattern.compile("%([i1-9]?)");
private final Queue<String> queue = new LinkedList<>();
private MiaoChat plugin = P.getPlugin();
private ChatConfig cc = plugin.getChatConfig();
public ChatListener() {
Bukkit.getPluginManager().registerEvents(this, P.instance);
new Statistics();
new SubscribeTask(true, true);
}
public void execute(final String commandList) {
execute(commandList, false);
}
public void execute(final String command, final boolean queued) {
final Iterator<? extends Player> it = C.Player.getOnlinePlayers().iterator();
if (it.hasNext()) {
final Player p = it.next();
p.sendPluginMessage(P.instance, MiaoMessage.CHANNEL, MiaoMessage.encode(command));
} else if (queued) {
queue.offer(command);
} else {
throw new RuntimeException("None player channel registered! Use queued");
}
}
private boolean processQueued() {
if (!queue.isEmpty()) {
final Iterator<? extends Player> it = C.Player.getOnlinePlayers().iterator();
if (!it.hasNext()) { return false; }
final Player p = it.next();
String command = queue.poll();
while (command != null) {
p.sendPluginMessage(P.instance, MiaoMessage.CHANNEL, MiaoMessage.encode(command));
command = queue.poll();
}
return true;
}
return false;
}
@EventHandler(priority = EventPriority.MONITOR, ignoreCancelled = true)
public void onChat(AsyncPlayerChatEvent e) {
Player p = e.getPlayer();
ChatRule cr = cc.getChatRule(e.getPlayer());
if (cr == null) {
// Log.d("玩家: %s 未发现可用ChatRule!", p.getName());
return;
}
e.setCancelled(true);
Tellraw tr = Tellraw.create();
handleChat(p, tr, cr, e.getMessage());
}
private void handleChat(Player p, Tellraw tr, ChatRule cr, String message) {
// Log.d("玩家: %s 使用 %s 规则 解析 %s", p.getName(), cr.getName(), message);
handleFormat(p, tr, cr);
handleTellraw(p, tr, cr, message);
handleSend(p, tr, cr.getRange());
}
private String handleFormat(Player p, Tellraw tr, ChatRule cr) {
LinkedList<String> formats = cr.getFormats();
// Log.d("处理前缀信息...");
for (String format : formats) {
ChatMessagePart cmp = cc.getFormat(format);
if (cmp != null) {
// Log.d("解析格式: %s", format);
cmp.then(tr, p);
} else {
// Log.d("追加文本: %s", format);
tr.then(PlaceholderAPI.setPlaceholders(p, format));
}
}
return ChatColor.getLastColors(formats.getLast());
}
private LinkedList<String> handleMessage(LinkedList<String> il, String message) {
LinkedList<String> mlist = new LinkedList<>();
// Log.d("处理聊天信息...");
if (!il.isEmpty()) {
for (String k : il) {
String[] args = message.split(k, 2);
if (!args[0].isEmpty()) {
// Log.d("追加文本: %s", args[0]);
mlist.add(args[0]);
}
// Log.d("解析物品: %s", args[0]);
mlist.add(k);
message = args[1];
}
}
if (!message.isEmpty()) {
// Log.d("追加文本: %s", message);
mlist.add(message);
}
return mlist;
}
private LinkedList<String> handlePattern(String message) {
Matcher m = ITEM_PATTERN.matcher(message);
Set<String> temp = new HashSet<>();
LinkedList<String> ilist = new LinkedList<>();
// Log.d("处理聊天物品信息...");
while (m.find()) {
String key = m.group(0);
if (key.length() == 2) {
if (temp.add(key)) {
// Log.d("解析物品关键词: %s", key);
ilist.add(key);
} else {
return null;
}
}
}
return ilist;
}
private void handleSend(Player p, Tellraw tr, int range) {
Collection<? extends Entity> plist;
if (range != 0) {
plist = p.getNearbyEntities(range, range, range);
tr.send(p);
} else {
plist = C.Player.getOnlinePlayers();
}
for (Entity ne : plist) {
if (ne instanceof Player && !offList.contains(((Player) ne).getName())) {
tr.send(ne);
}
}
Bukkit.getConsoleSender().sendMessage(tr.toOldMessageFormat());
}
private void handleTellraw(Player player, Tellraw tr, ChatRule cr, String message) {
if (message.isEmpty()) { return; }
if (player.hasPermission("MiaoChat.color")) {
message = ChatColor.translateAlternateColorCodes('&', message);
}
if (!cr.isItem()) {
tr.then(message);
return;
}
LinkedList<String> il = handlePattern(message);
// 如果返回null说明存在相同的物品
if (il == null) {
Log.toSender(player, "§c不允许展示相同的物品!");
return;
}
LinkedList<String> ml = handleMessage(il, message);
// Log.d("处理Tellraw格式...");
while (!ml.isEmpty()) {
String mm = ml.removeFirst();
if (il.contains(mm)) {
char k = mm.charAt(1);
ItemStack is = k == 'i' ? player.getItemInHand() : player.getInventory().getItem(k - '0' - 1);
if (is != null && is.getType() != Material.AIR) {
// Log.d("处理物品: %s", mm);
tr.then(String.format(ChatColor.translateAlternateColorCodes('&', cr.getItemformat()), L10N.getName(is)));
tr.item(is);
}
} else {
// Log.d("追加聊天: %s", mm);
tr.then(cr.getLastColor() + mm);
}
}
}
}