Towny event broadcasting to Discord and masked links #96

Merged
NorbiPeti merged 7 commits from dev into master 2019-01-20 21:39:54 +00:00
27 changed files with 1046 additions and 1100 deletions
Showing only changes of commit 3f387a9775 - Show all commits

View file

@ -1,25 +0,0 @@
package buttondevteam.chat;
import org.bukkit.Bukkit;
public class AnnouncerThread implements Runnable {
private static int AnnounceMessageIndex = 0;
@Override
public void run() {
while (!PluginMain.Instance.stop) {
try {
Thread.sleep(PluginMain.AnnounceTime);
} catch (InterruptedException ex) {
Thread.currentThread().interrupt();
}
if (Bukkit.getOnlinePlayers().size() == 0) continue; //Don't post to Discord if nobody is on
if (PluginMain.AnnounceMessages.size() > AnnounceMessageIndex) {
Bukkit.broadcastMessage(PluginMain.AnnounceMessages.get(AnnounceMessageIndex));
AnnounceMessageIndex++;
if (AnnounceMessageIndex == PluginMain.AnnounceMessages.size())
AnnounceMessageIndex = 0;
}
}
}
}

View file

@ -1,135 +1,132 @@
package buttondevteam.chat; package buttondevteam.chat;
import buttondevteam.lib.chat.Color; import buttondevteam.chat.components.flair.FlairStates;
import buttondevteam.lib.player.EnumPlayerData; import buttondevteam.lib.chat.Color;
import buttondevteam.lib.player.PlayerClass; import buttondevteam.lib.player.EnumPlayerData;
import buttondevteam.lib.player.PlayerData; import buttondevteam.lib.player.PlayerClass;
import buttondevteam.lib.player.TBMCPlayerBase; import buttondevteam.lib.player.PlayerData;
import org.bukkit.Bukkit; import buttondevteam.lib.player.TBMCPlayerBase;
import org.bukkit.Location; import org.bukkit.Bukkit;
import org.bukkit.entity.Player; import org.bukkit.entity.Player;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.List; import java.util.List;
@PlayerClass(pluginname = "Button1Chat") @PlayerClass(pluginname = "Button1Chat")
public class ChatPlayer extends TBMCPlayerBase { public class ChatPlayer extends TBMCPlayerBase {
public PlayerData<String> UserName() { public PlayerData<String> UserName() {
return data(null); return data(null);
} }
public List<String> UserNames() { public List<String> UserNames() {
return data(new ArrayList<String>()).get(); return data(new ArrayList<String>()).get();
} }
public PlayerData<Integer> FlairTime() { public PlayerData<Integer> FlairTime() {
return data(FlairTimeNone); return data(FlairTimeNone);
} }
public EnumPlayerData<FlairStates> FlairState() { public EnumPlayerData<FlairStates> FlairState() {
return dataEnum(FlairStates.class, FlairStates.NoComment); return dataEnum(FlairStates.class, FlairStates.NoComment);
} }
public PlayerData<Integer> FCount() { public PlayerData<Integer> FCount() {
return data(0); return data(0);
} }
public PlayerData<Integer> FDeaths() { public PlayerData<Integer> FDeaths() {
return data(0); return data(0);
} }
public PlayerData<Boolean> FlairCheater() { public PlayerData<Boolean> FlairCheater() {
return data(false); return data(false);
} }
public PlayerData<ArrayList<Integer>> NameColorLocations() { // No byte[], no TIntArrayList public PlayerData<ArrayList<Integer>> NameColorLocations() { // No byte[], no TIntArrayList
return data(null); return data(null);
} }
public Location SavedLocation; public boolean Working;
public boolean Working; // public int Tables = 10;
// public int Tables = 10; public boolean RainbowPresserColorMode = false;
public boolean SendingLink = false; public Color OtherColorMode = null;
public boolean RainbowPresserColorMode = false; public boolean ChatOnly = false;
public Color OtherColorMode = null;
public boolean ChatOnly = false; public static final int FlairTimeNonPresser = -1;
public int LoginWarningCount = 0; public static final int FlairTimeCantPress = -2;
public static final int FlairTimeNone = -3;
public static final int FlairTimeNonPresser = -1;
public static final int FlairTimeCantPress = -2; /**
public static final int FlairTimeNone = -3; * Gets the player's flair, optionally formatting for Minecraft.
*
/** * @param noformats
* Gets the player's flair, optionally formatting for Minecraft. * The MC formatting codes will be only applied if false
* * @return The flair
* @param noformats */
* The MC formatting codes will be only applied if false public String GetFormattedFlair(boolean noformats) {
* @return The flair int time = FlairTime().get();
*/ if (time == FlairTimeCantPress)
public String GetFormattedFlair(boolean noformats) { return noformats ? "(can't press)" : "§r(--s)§r";
int time = FlairTime().get(); if (time == FlairTimeNonPresser)
if (time == FlairTimeCantPress) return noformats ? "(non-presser)" : "§7(--s)§r";
return noformats ? "(can't press)" : "§r(--s)§r"; if (time == FlairTimeNone)
if (time == FlairTimeNonPresser) return "";
return noformats ? "(non-presser)" : "§7(--s)§r"; return noformats ? String.format("(%ds)", time) : String.format("§%x(%ds)§r", GetFlairColor(), time);
if (time == FlairTimeNone) }
return "";
return noformats ? String.format("(%ds)", time) : String.format("§%x(%ds)§r", GetFlairColor(), time); /**
} * Gets the player's flair, formatted for Minecraft.
*
/** * @return The flair
* Gets the player's flair, formatted for Minecraft. */
* public String GetFormattedFlair() {
* @return The flair return GetFormattedFlair(false);
*/ }
public String GetFormattedFlair() {
return GetFormattedFlair(false); public void SetFlair(int time) {
} FlairTime().set(time);
FlairUpdate();
public void SetFlair(int time) { }
FlairTime().set(time);
FlairUpdate(); public void SetFlair(int time, boolean cheater) {
} FlairTime().set(time);
FlairCheater().set(cheater);
public void SetFlair(int time, boolean cheater) { FlairUpdate();
FlairTime().set(time); }
FlairCheater().set(cheater);
FlairUpdate(); public void FlairUpdate() {
}
// Flairs from Command Block The Button - Teams
public void FlairUpdate() { // PluginMain.Instance.getServer().getScoreboardManager().getMainScoreboard().getTeams().add()
Player p = Bukkit.getPlayer(uuid);
// Flairs from Command Block The Button - Teams if (p != null)
// PluginMain.Instance.getServer().getScoreboardManager().getMainScoreboard().getTeams().add() p.setPlayerListName(String.format("%s%s", p.getDisplayName(), GetFormattedFlair()));
Player p = Bukkit.getPlayer(uuid); }
if (p != null)
p.setPlayerListName(String.format("%s%s", p.getDisplayName(), GetFormattedFlair())); public short GetFlairColor() {
} if (FlairCheater().get())
return 0x5;
public short GetFlairColor() { final int flairTime = FlairTime().get();
if (FlairCheater().get()) if (flairTime == FlairTimeNonPresser)
return 0x5; return 0x7;
final int flairTime = FlairTime().get(); else if (flairTime == FlairTimeCantPress)
if (flairTime == FlairTimeNonPresser) return 0xf;
return 0x7; else if (flairTime <= 60 && flairTime >= 52)
else if (flairTime == FlairTimeCantPress) return 0x5;
return 0xf; else if (flairTime <= 51 && flairTime >= 42)
else if (flairTime <= 60 && flairTime >= 52) return 0x9;
return 0x5; else if (flairTime <= 41 && flairTime >= 32)
else if (flairTime <= 51 && flairTime >= 42) return 0xa;
return 0x9; else if (flairTime <= 31 && flairTime >= 22)
else if (flairTime <= 41 && flairTime >= 32) return 0xe;
return 0xa; else if (flairTime <= 21 && flairTime >= 11)
else if (flairTime <= 31 && flairTime >= 22) return 0x6;
return 0xe; else if (flairTime <= 11 && flairTime >= 0)
else if (flairTime <= 21 && flairTime >= 11) return 0xc;
return 0x6; return 0x00; //Return 0 if none or too high, so names will get aqua default color, not white
else if (flairTime <= 11 && flairTime >= 0) }
return 0xc;
return 0x00; //Return 0 if none or too high, so names will get aqua default color, not white public double getF() {
} return (double) FCount().get() / (double) FDeaths().get();
}
public double getF() { }
return (double) FCount().get() / (double) FDeaths().get();
}
}

View file

@ -1,90 +0,0 @@
package buttondevteam.chat;
import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.URL;
import java.net.URLDecoder;
import java.util.Enumeration;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;
public class JarUtils {
public static boolean extractFromJar(final String fileName,
final String dest) throws IOException {
if (getRunningJar() == null) {
return false;
}
final File file = new File(dest);
if (file.isDirectory()) {
file.mkdir();
return false;
}
if (!file.exists()) {
file.getParentFile().mkdirs();
}
final JarFile jar = getRunningJar();
final Enumeration<JarEntry> e = jar.entries();
while (e.hasMoreElements()) {
final JarEntry je = e.nextElement();
if (!je.getName().contains(fileName)) {
continue;
}
final InputStream in = new BufferedInputStream(
jar.getInputStream(je));
final OutputStream out = new BufferedOutputStream(
new FileOutputStream(file));
copyInputStream(in, out);
jar.close();
return true;
}
jar.close();
return false;
}
private final static void copyInputStream(final InputStream in,
final OutputStream out) throws IOException {
try {
final byte[] buff = new byte[4096];
int n;
while ((n = in.read(buff)) > 0) {
out.write(buff, 0, n);
}
} finally {
out.flush();
out.close();
in.close();
}
}
public static URL getJarUrl(final File file) throws IOException {
return new URL("jar:" + file.toURI().toURL().toExternalForm() + "!/");
}
public static JarFile getRunningJar() throws IOException {
if (!RUNNING_FROM_JAR) {
return null; // null if not running from jar
}
String path = new File(JarUtils.class.getProtectionDomain()
.getCodeSource().getLocation().getPath()).getAbsolutePath();
path = URLDecoder.decode(path, "UTF-8");
return new JarFile(path);
}
private static boolean RUNNING_FROM_JAR = false;
static {
final URL resource = JarUtils.class.getClassLoader().getResource(
"plugin.yml");
if (resource != null) {
RUNNING_FROM_JAR = true;
}
}
}

View file

@ -3,52 +3,36 @@ package buttondevteam.chat;
import buttondevteam.chat.commands.YeehawCommand; import buttondevteam.chat.commands.YeehawCommand;
import buttondevteam.chat.commands.ucmds.TownColorCommand; import buttondevteam.chat.commands.ucmds.TownColorCommand;
import buttondevteam.chat.components.TownColorComponent; import buttondevteam.chat.components.TownColorComponent;
import buttondevteam.chat.components.TownyComponent; import buttondevteam.chat.components.announce.AnnouncerComponent;
import buttondevteam.chat.components.flair.FlairComponent;
import buttondevteam.chat.components.towny.TownyComponent;
import buttondevteam.chat.listener.PlayerJoinLeaveListener; import buttondevteam.chat.listener.PlayerJoinLeaveListener;
import buttondevteam.chat.listener.PlayerListener; import buttondevteam.chat.listener.PlayerListener;
import buttondevteam.chat.listener.TownyListener; import buttondevteam.chat.listener.TownyListener;
import buttondevteam.component.channel.Channel; import buttondevteam.component.channel.Channel;
import buttondevteam.component.channel.Channel.RecipientTestResult;
import buttondevteam.lib.TBMCCoreAPI; import buttondevteam.lib.TBMCCoreAPI;
import buttondevteam.lib.architecture.Component; import buttondevteam.lib.architecture.Component;
import buttondevteam.lib.chat.Color; import buttondevteam.lib.chat.Color;
import buttondevteam.lib.chat.TBMCChatAPI; import buttondevteam.lib.chat.TBMCChatAPI;
import buttondevteam.lib.player.TBMCPlayerBase;
import com.earth2me.essentials.Essentials; import com.earth2me.essentials.Essentials;
import com.google.gson.JsonArray;
import com.google.gson.JsonObject;
import com.google.gson.JsonParser;
import com.palmergames.bukkit.towny.Towny;
import com.palmergames.bukkit.towny.exceptions.NotRegisteredException;
import com.palmergames.bukkit.towny.object.Nation;
import com.palmergames.bukkit.towny.object.Resident;
import com.palmergames.bukkit.towny.object.Town;
import com.palmergames.bukkit.towny.object.TownyUniverse;
import lombok.val; import lombok.val;
import net.milkbowl.vault.chat.Chat; import net.milkbowl.vault.chat.Chat;
import net.milkbowl.vault.economy.Economy; import net.milkbowl.vault.economy.Economy;
import net.milkbowl.vault.permission.Permission; import net.milkbowl.vault.permission.Permission;
import org.bukkit.Bukkit; import org.bukkit.Bukkit;
import org.bukkit.command.CommandSender;
import org.bukkit.command.ConsoleCommandSender; import org.bukkit.command.ConsoleCommandSender;
import org.bukkit.configuration.file.YamlConfiguration; import org.bukkit.configuration.file.YamlConfiguration;
import org.bukkit.entity.Player;
import org.bukkit.plugin.RegisteredServiceProvider; import org.bukkit.plugin.RegisteredServiceProvider;
import org.bukkit.plugin.java.JavaPlugin; import org.bukkit.plugin.java.JavaPlugin;
import org.bukkit.scoreboard.Scoreboard; import org.bukkit.scoreboard.Scoreboard;
import org.dynmap.towny.DTBridge; import org.dynmap.towny.DTBridge;
import org.dynmap.towny.DynmapTownyPlugin; import org.dynmap.towny.DynmapTownyPlugin;
import org.htmlcleaner.HtmlCleaner;
import org.htmlcleaner.TagNode;
import java.io.File; import java.io.File;
import java.io.IOException; import java.util.Arrays;
import java.io.InputStream; import java.util.HashMap;
import java.net.URL; import java.util.List;
import java.net.URLConnection; import java.util.Map;
import java.net.UnknownHostException;
import java.text.SimpleDateFormat;
import java.util.*;
import java.util.function.Function; import java.util.function.Function;
import java.util.stream.Collectors; import java.util.stream.Collectors;
@ -57,26 +41,11 @@ public class PluginMain extends JavaPlugin { // Translated to Java: 2015.07.15.
// https://www.reddit.com/r/thebutton/comments/31c32v/i_pressed_the_button_without_really_thinking/ // https://www.reddit.com/r/thebutton/comments/31c32v/i_pressed_the_button_without_really_thinking/
public static PluginMain Instance; public static PluginMain Instance;
public static ConsoleCommandSender Console; public static ConsoleCommandSender Console;
private final static String FlairThreadURL = "https://www.reddit.com/r/Chromagamers/comments/51ys94/flair_thread_for_the_mc_server/";
public static Scoreboard SB; public static Scoreboard SB;
public static TownyUniverse TU;
private static ArrayList<String> Towns;
private static ArrayList<String> Nations;
public static Channel TownChat; public static Channel TownChat;
public static Channel NationChat; public static Channel NationChat;
private static Channel RPChannel;
/**
* <p>
* This variable is used as a cache for flair state checking when reading the flair thread.
* </p>
* <p>
* It's used because normally it has to load all associated player files every time to read the flair state
* </p>
*/
private Set<String> PlayersWithFlairs = new HashSet<>();
// Fired when plugin is first enabled // Fired when plugin is first enabled
@Override @Override
@ -92,18 +61,12 @@ public class PluginMain extends JavaPlugin { // Translated to Java: 2015.07.15.
LoadFiles(); LoadFiles();
SB = getServer().getScoreboardManager().getMainScoreboard(); // Main can be detected with @a[score_...] SB = getServer().getScoreboardManager().getMainScoreboard(); // Main can be detected with @a[score_...]
TU = ((Towny) Bukkit.getPluginManager().getPlugin("Towny")).getTownyUniverse();
Towns = TU.getTownsMap().values().stream().map(Town::getName).collect(Collectors.toCollection(ArrayList::new)); // Creates a snapshot of towns, new towns will be added when needed
Nations = TU.getNationsMap().values().stream().map(Nation::getName).collect(Collectors.toCollection(ArrayList::new)); // Same here but with nations
TownColors.keySet().removeIf(t -> !TU.getTownsMap().containsKey(t)); // Removes town colors for deleted/renamed towns Component.registerComponent(this, new TownyComponent());
NationColor.keySet().removeIf(n -> !TU.getNationsMap().containsKey(n)); // Removes nation colors for deleted/renamed nations TownColors.keySet().removeIf(t -> !TownyComponent.TU.getTownsMap().containsKey(t)); // Removes town colors for deleted/renamed towns
NationColor.keySet().removeIf(n -> !TownyComponent.TU.getNationsMap().containsKey(n)); // Removes nation colors for deleted/renamed nations
TBMCChatAPI.RegisterChatChannel( TBMCChatAPI.RegisterChatChannel(new Channel("§7RP§f", Color.Gray, "rp", null)); //Since it's null, it's recognised as global
TownChat = new Channel("§3TC§f", Color.DarkAqua, "tc", s -> checkTownNationChat(s, false)));
TBMCChatAPI.RegisterChatChannel(
NationChat = new Channel("§6NC§f", Color.Gold, "nc", s -> checkTownNationChat(s, true)));
TBMCChatAPI.RegisterChatChannel(RPChannel = new Channel("§7RP§f", Color.Gray, "rp", null)); //Since it's null, it's recognised as global
Bukkit.getScheduler().runTask(this, () -> { Bukkit.getScheduler().runTask(this, () -> {
val dtp = (DynmapTownyPlugin) Bukkit.getPluginManager().getPlugin("Dynmap-Towny"); val dtp = (DynmapTownyPlugin) Bukkit.getPluginManager().getPlugin("Dynmap-Towny");
@ -116,11 +79,9 @@ public class PluginMain extends JavaPlugin { // Translated to Java: 2015.07.15.
if (!setupEconomy() || !setupPermissions()) if (!setupEconomy() || !setupPermissions())
TBMCCoreAPI.SendException("We're in trouble", new Exception("Failed to set up economy or permissions!")); TBMCCoreAPI.SendException("We're in trouble", new Exception("Failed to set up economy or permissions!"));
new Thread(this::FlairGetterThreadMethod).start();
new Thread(new AnnouncerThread()).start();
Component.registerComponent(this, new TownyComponent());
Component.registerComponent(this, new TownColorComponent()); Component.registerComponent(this, new TownColorComponent());
Component.registerComponent(this, new FlairComponent()); //The original purpose of this plugin
Component.registerComponent(this, new AnnouncerComponent());
} }
/** /**
@ -140,152 +101,14 @@ public class PluginMain extends JavaPlugin { // Translated to Java: 2015.07.15.
} }
} }
public Boolean stop = false;
public static Essentials essentials = null; public static Essentials essentials = null;
// Fired when plugin is disabled // Fired when plugin is disabled
@Override @Override
public void onDisable() { public void onDisable() {
SaveFiles(); SaveFiles();
stop = true;
} }
private void FlairGetterThreadMethod() {
int errorcount = 0;
while (!stop) {
try {
String body = TBMCCoreAPI.DownloadString(FlairThreadURL + ".json?limit=1000");
JsonArray json = new JsonParser().parse(body).getAsJsonArray().get(1).getAsJsonObject().get("data")
.getAsJsonObject().get("children").getAsJsonArray();
for (Object obj : json) {
JsonObject item = (JsonObject) obj;
String author = item.get("data").getAsJsonObject().get("author").getAsString();
String ign = item.get("data").getAsJsonObject().get("body").getAsString();
int start = ign.indexOf("IGN:") + "IGN:".length();
if (start == -1 + "IGN:".length())
continue;
int end = ign.indexOf(' ', start);
if (end == -1 || end == start)
end = ign.indexOf('\n', start);
if (end == -1 || end == start)
ign = ign.substring(start);
else
ign = ign.substring(start, end);
ign = ign.trim();
if (PlayersWithFlairs.contains(ign))
continue;
try (ChatPlayer mp = TBMCPlayerBase.getFromName(ign, ChatPlayer.class)) { // Loads player file
if (mp == null)
continue;
/*
* if (!JoinedBefore(mp, 2015, 6, 5)) continue;
*/
if (!mp.UserNames().contains(author))
mp.UserNames().add(author);
if (mp.FlairState().get().equals(FlairStates.NoComment)) {
mp.FlairState().set(FlairStates.Commented);
ConfirmUserMessage(mp);
}
PlayersWithFlairs.add(ign); // Don't redownload even if flair isn't accepted
}
}
} catch (Exception e) {
errorcount++;
if (errorcount >= 10) {
errorcount = 0;
if (!e.getMessage().contains("Server returned HTTP response code")
&& !(e instanceof UnknownHostException))
TBMCCoreAPI.SendException("Error while getting flairs from Reddit!", e);
}
}
try {
Thread.sleep(10000);
} catch (InterruptedException ex) {
Thread.currentThread().interrupt();
}
}
}
public void DownloadFlair(ChatPlayer mp) throws IOException {
String[] flairdata = TBMCCoreAPI
.DownloadString("http://karmadecay.com/thebutton-data.php?users=" + mp.UserName().get())
.replace("\"", "").split(":");
String flair;
if (flairdata.length > 1)
flair = flairdata[1];
else
flair = "";
String flairclass;
if (flairdata.length > 2)
flairclass = flairdata[2];
else
flairclass = "unknown";
SetFlair(mp, flair, flairclass, mp.UserName().get());
}
private void SetFlair(ChatPlayer p, String text, String flairclass, String username) {
p.UserName().set(username);
p.FlairState().set(FlairStates.Recognised);
switch (flairclass) {
case "cheater":
p.SetFlair(Short.parseShort(text), true);
return;
case "unknown":
try {
if (CheckForJoinDate(p)) {
if (text.equals("-1")) // If true, only non-presser/can't press; if false, any flair (but we can still detect can't press)
p.SetFlair(ChatPlayer.FlairTimeNonPresser);
else
p.SetFlair(ChatPlayer.FlairTimeNone); // Flair unknown
} else {
p.SetFlair(ChatPlayer.FlairTimeCantPress);
}
} catch (Exception e) {
p.FlairState().set(FlairStates.Commented); // Flair unknown
p.SetFlair(ChatPlayer.FlairTimeNone);
TBMCCoreAPI.SendException("Error while checking join date for player " + p.PlayerName() + "!", e);
}
return;
default:
break;
}
p.SetFlair(Short.parseShort(text));
}
private static boolean CheckForJoinDate(ChatPlayer mp) throws Exception {
return JoinedBefore(mp, 2015, 4, 1);
}
private static boolean JoinedBefore(ChatPlayer mp, int year, int month, int day) throws Exception {
URL url = new URL("https://www.reddit.com/u/" + mp.UserName());
URLConnection con = url.openConnection();
con.setRequestProperty("User-Agent", "TheButtonAutoFlair");
InputStream in = con.getInputStream();
HtmlCleaner cleaner = new HtmlCleaner();
TagNode node = cleaner.clean(in);
node = node.getElementsByAttValue("class", "age", true, true)[0];
node = node.getElementsByName("time", false)[0];
String joindate = node.getAttributeByName("datetime");
SimpleDateFormat parserSDF = new SimpleDateFormat("yyyy-MM-dd");
joindate = joindate.split("T")[0];
Date date = parserSDF.parse(joindate);
return date.before(new Calendar.Builder().setTimeZone(TimeZone.getTimeZone("UTC")).setDate(year, month, day)
.build().getTime());
}
public static void ConfirmUserMessage(ChatPlayer mp) {
Player p = Bukkit.getPlayer(mp.getUUID());
if (mp.FlairState().get().equals(FlairStates.Commented) && p != null)
if (mp.UserNames().size() > 1)
p.sendMessage(
"§9Multiple Reddit users commented your name. You can select with /u accept.§r §6Type /u accept or /u ignore§r");
else
p.sendMessage("§9A Reddit user commented your name. Is that you?§r §6Type /u accept or /u ignore§r");
}
public static ArrayList<String> AnnounceMessages = new ArrayList<>();
public static int AnnounceTime = 15 * 60 * 1000;
/** /**
* Names lowercased * Names lowercased
*/ */
@ -305,9 +128,6 @@ public class PluginMain extends JavaPlugin { // Translated to Java: 2015.07.15.
yc.load(file); yc.load(file);
PlayerListener.NotificationSound = yc.getString("notificationsound"); PlayerListener.NotificationSound = yc.getString("notificationsound");
PlayerListener.NotificationPitch = yc.getDouble("notificationpitch"); PlayerListener.NotificationPitch = yc.getDouble("notificationpitch");
AnnounceTime = yc.getInt("announcetime", 15 * 60 * 1000);
AnnounceMessages.addAll(yc.getStringList("announcements"));
PlayerListener.AlphaDeaths = yc.getInt("alphadeaths");
val cs = yc.getConfigurationSection("towncolors"); val cs = yc.getConfigurationSection("towncolors");
if (cs != null) if (cs != null)
TownColors.putAll(cs.getValues(true).entrySet().stream() TownColors.putAll(cs.getValues(true).entrySet().stream()
@ -333,9 +153,6 @@ public class PluginMain extends JavaPlugin { // Translated to Java: 2015.07.15.
YamlConfiguration yc = new YamlConfiguration(); YamlConfiguration yc = new YamlConfiguration();
yc.set("notificationsound", PlayerListener.NotificationSound); yc.set("notificationsound", PlayerListener.NotificationSound);
yc.set("notificationpitch", PlayerListener.NotificationPitch); yc.set("notificationpitch", PlayerListener.NotificationPitch);
yc.set("announcetime", AnnounceTime);
yc.set("announcements", AnnounceMessages);
yc.set("alphadeaths", PlayerListener.AlphaDeaths);
yc.createSection("towncolors", TownColors.entrySet().stream().collect(Collectors.toMap(Map.Entry::getKey, yc.createSection("towncolors", TownColors.entrySet().stream().collect(Collectors.toMap(Map.Entry::getKey,
v -> Arrays.stream(v.getValue()).map(Enum::toString).toArray(String[]::new)))); v -> Arrays.stream(v.getValue()).map(Enum::toString).toArray(String[]::new))));
yc.set("towncolorcount", TownColorCommand.ColorCount); yc.set("towncolorcount", TownColorCommand.ColorCount);
@ -361,16 +178,6 @@ public class PluginMain extends JavaPlugin { // Translated to Java: 2015.07.15.
return (permission != null); return (permission != null);
} }
private boolean setupChat() {
RegisteredServiceProvider<Chat> chatProvider = getServer().getServicesManager()
.getRegistration(net.milkbowl.vault.chat.Chat.class);
if (chatProvider != null) {
chat = chatProvider.getProvider();
}
return (chat != null);
}
private boolean setupEconomy() { private boolean setupEconomy() {
RegisteredServiceProvider<Economy> economyProvider = getServer().getServicesManager() RegisteredServiceProvider<Economy> economyProvider = getServer().getServicesManager()
.getRegistration(net.milkbowl.vault.economy.Economy.class); .getRegistration(net.milkbowl.vault.economy.Economy.class);
@ -381,53 +188,4 @@ public class PluginMain extends JavaPlugin { // Translated to Java: 2015.07.15.
return (economy != null); return (economy != null);
} }
/**
* Return the error message for the message sender if they can't send it and the score
*/
private static RecipientTestResult checkTownNationChat(CommandSender sender, boolean nationchat) {
if (!(sender instanceof Player))
return new RecipientTestResult("§cYou are not a player!");
Resident resident = PluginMain.TU.getResidentMap().get(sender.getName().toLowerCase());
RecipientTestResult result = checkTownNationChatInternal(sender, nationchat, resident);
if (result.errormessage != null && resident != null && resident.getModes().contains("spy")) // Only use spy if they wouldn't see it
result = new RecipientTestResult(1000, "allspies"); // There won't be more than a thousand towns/nations probably
return result;
}
private static RecipientTestResult checkTownNationChatInternal(CommandSender sender, boolean nationchat,
Resident resident) {
try {
/*
* p.sendMessage(String.format("[SPY-%s] - %s: %s", channel.DisplayName, ((Player) sender).getDisplayName(), message));
*/
Town town = null;
if (resident != null && resident.hasTown())
town = resident.getTown();
if (town == null)
return new RecipientTestResult("You aren't in a town.");
Nation nation = null;
int index;
if (nationchat) {
if (town.hasNation())
nation = town.getNation();
if (nation == null)
return new RecipientTestResult("Your town isn't in a nation.");
index = getTownNationIndex(nation.getName(), true);
} else
index = getTownNationIndex(town.getName(), false);
return new RecipientTestResult(index, nationchat ? nation.getName() : town.getName());
} catch (NotRegisteredException e) {
return new RecipientTestResult("You (probably) aren't knwon by Towny! (Not in a town)");
}
}
public static int getTownNationIndex(String name, boolean nation) {
val list = nation ? Nations : Towns;
int index = list.indexOf(name);
if (index < 0) {
list.add(name);
index = list.size() - 1;
}
return index;
}
} }

View file

@ -39,9 +39,9 @@ public class HistoryCommand extends UCommandBase {
sender.sendMessage("§6---- Chat History ----"); sender.sendMessage("§6---- Chat History ----");
Stream<Channel> stream; Stream<Channel> stream;
if (args.length == 0) { if (args.length == 0) {
stream = Channel.getChannels().stream(); stream = Channel.getChannels();
} else { } else {
Optional<Channel> och = Channel.getChannels().stream().filter(chan -> chan.ID.equalsIgnoreCase(args[0])).findAny(); Optional<Channel> och = Channel.getChannels().filter(chan -> chan.ID.equalsIgnoreCase(args[0])).findAny();
if (!och.isPresent()) { if (!och.isPresent()) {
sender.sendMessage("§cChannel not found. Use the ID, for example: /" + (hc == null ? "u history" : hc.GetCommandPath()) + " ooc"); sender.sendMessage("§cChannel not found. Use the ID, for example: /" + (hc == null ? "u history" : hc.GetCommandPath()) + " ooc");
return true; return true;

View file

@ -2,6 +2,7 @@ package buttondevteam.chat.commands.ucmds;
import buttondevteam.chat.ChatPlayer; import buttondevteam.chat.ChatPlayer;
import buttondevteam.chat.PluginMain; import buttondevteam.chat.PluginMain;
import buttondevteam.chat.components.towny.TownyComponent;
import buttondevteam.chat.listener.PlayerJoinLeaveListener; import buttondevteam.chat.listener.PlayerJoinLeaveListener;
import buttondevteam.lib.chat.Color; import buttondevteam.lib.chat.Color;
import buttondevteam.lib.chat.CommandClass; import buttondevteam.lib.chat.CommandClass;
@ -34,7 +35,7 @@ public class NColorCommand extends UCommandBase {
Resident res; Resident res;
Town town; Town town;
try { try {
if ((res = PluginMain.TU.getResidentMap().get(player.getName().toLowerCase())) == null || !res.hasTown() if ((res = TownyComponent.TU.getResidentMap().get(player.getName().toLowerCase())) == null || !res.hasTown()
|| (town = res.getTown()) == null) { || (town = res.getTown()) == null) {
player.sendMessage("§cYou need to be in a town."); player.sendMessage("§cYou need to be in a town.");
return true; return true;

View file

@ -1,6 +1,6 @@
package buttondevteam.chat.commands.ucmds; package buttondevteam.chat.commands.ucmds;
import buttondevteam.chat.PluginMain; import buttondevteam.chat.components.towny.TownyComponent;
import buttondevteam.lib.TBMCCoreAPI; import buttondevteam.lib.TBMCCoreAPI;
import buttondevteam.lib.chat.CommandClass; import buttondevteam.lib.chat.CommandClass;
import buttondevteam.lib.chat.OptionallyPlayerCommandClass; import buttondevteam.lib.chat.OptionallyPlayerCommandClass;
@ -26,8 +26,8 @@ public class NationColorCommand extends UCommandBase {
@Override @Override
public boolean OnCommand(Player player, String alias, String[] args) { public boolean OnCommand(Player player, String alias, String[] args) {
Resident res; Resident res;
if (!(PluginMain.TU.getResidentMap().containsKey(player.getName().toLowerCase()) if (!(TownyComponent.TU.getResidentMap().containsKey(player.getName().toLowerCase())
&& (res = PluginMain.TU.getResidentMap().get(player.getName().toLowerCase())).isKing())) { && (res = TownyComponent.TU.getResidentMap().get(player.getName().toLowerCase())).isKing())) {
player.sendMessage("§cYou need to be the king of a nation to set it's colors."); player.sendMessage("§cYou need to be the king of a nation to set it's colors.");
return true; return true;
} }

View file

@ -1,6 +1,6 @@
package buttondevteam.chat.commands.ucmds; package buttondevteam.chat.commands.ucmds;
import buttondevteam.chat.PluginMain; import buttondevteam.chat.components.towny.TownyComponent;
import buttondevteam.lib.TBMCCoreAPI; import buttondevteam.lib.TBMCCoreAPI;
import buttondevteam.lib.chat.CommandClass; import buttondevteam.lib.chat.CommandClass;
import buttondevteam.lib.chat.OptionallyPlayerCommandClass; import buttondevteam.lib.chat.OptionallyPlayerCommandClass;
@ -31,8 +31,8 @@ public class TownColorCommand extends UCommandBase {
@Override @Override
public boolean OnCommand(Player player, String alias, String[] args) { public boolean OnCommand(Player player, String alias, String[] args) {
Resident res; Resident res;
if (!(PluginMain.TU.getResidentMap().containsKey(player.getName().toLowerCase()) if (!(TownyComponent.TU.getResidentMap().containsKey(player.getName().toLowerCase())
&& (res = PluginMain.TU.getResidentMap().get(player.getName().toLowerCase())).isMayor())) { && (res = TownyComponent.TU.getResidentMap().get(player.getName().toLowerCase())).isMayor())) {
player.sendMessage("§cYou need to be the mayor of a town to set it's colors."); player.sendMessage("§cYou need to be the mayor of a town to set it's colors.");
return true; return true;
} }

View file

@ -1,6 +1,7 @@
package buttondevteam.chat.commands.ucmds.admin; package buttondevteam.chat.commands.ucmds.admin;
import buttondevteam.chat.PluginMain; import buttondevteam.chat.PluginMain;
import buttondevteam.chat.components.towny.TownyComponent;
import buttondevteam.chat.listener.TownyListener; import buttondevteam.chat.listener.TownyListener;
import buttondevteam.lib.chat.Color; import buttondevteam.lib.chat.Color;
import com.palmergames.bukkit.towny.object.Nation; import com.palmergames.bukkit.towny.object.Nation;
@ -31,7 +32,7 @@ public class NationColorCommand extends AdminCommandBase {
sender.sendMessage("§cYou can only use one color as a nation color."); sender.sendMessage("§cYou can only use one color as a nation color.");
return true; return true;
} }
final Nation nation = PluginMain.TU.getNationsMap().get(args[0].toLowerCase()); final Nation nation = TownyComponent.TU.getNationsMap().get(args[0].toLowerCase());
if (nation == null) { if (nation == null) {
sender.sendMessage("§cThe nation '" + args[0] + "' cannot be found."); sender.sendMessage("§cThe nation '" + args[0] + "' cannot be found.");
return true; return true;

View file

@ -1,6 +1,7 @@
package buttondevteam.chat.commands.ucmds.admin; package buttondevteam.chat.commands.ucmds.admin;
import buttondevteam.chat.PluginMain; import buttondevteam.chat.PluginMain;
import buttondevteam.chat.components.towny.TownyComponent;
import buttondevteam.chat.listener.TownyListener; import buttondevteam.chat.listener.TownyListener;
import buttondevteam.lib.chat.Color; import buttondevteam.lib.chat.Color;
import com.palmergames.bukkit.towny.object.Town; import com.palmergames.bukkit.towny.object.Town;
@ -35,12 +36,12 @@ public class TownColorCommand extends AdminCommandBase {
public static boolean SetTownColor(CommandSender sender, String alias, String[] args) { public static boolean SetTownColor(CommandSender sender, String alias, String[] args) {
if (args.length < 2) if (args.length < 2)
return false; return false;
if (!PluginMain.TU.getTownsMap().containsKey(args[0].toLowerCase())) { if (!TownyComponent.TU.getTownsMap().containsKey(args[0].toLowerCase())) {
sender.sendMessage("§cThe town '" + args[0] + "' cannot be found."); sender.sendMessage("§cThe town '" + args[0] + "' cannot be found.");
return true; return true;
} }
Color[] clrs = new Color[args.length - 1]; Color[] clrs = new Color[args.length - 1];
Town targetTown = PluginMain.TU.getTownsMap().get(args[0].toLowerCase()); Town targetTown = TownyComponent.TU.getTownsMap().get(args[0].toLowerCase());
for (int i = 1; i < args.length; i++) { for (int i = 1; i < args.length; i++) {
val c = getColorOrSendError(args[i], sender); val c = getColorOrSendError(args[i], sender);
if (!c.isPresent()) if (!c.isPresent())
@ -50,7 +51,7 @@ public class TownColorCommand extends AdminCommandBase {
for (Map.Entry<String, Color[]> other : PluginMain.TownColors.entrySet()) { for (Map.Entry<String, Color[]> other : PluginMain.TownColors.entrySet()) {
Color nc, tnc; Color nc, tnc;
try { try {
nc = PluginMain.NationColor.get(PluginMain.TU.getTownsMap().get(other.getKey()).getNation().getName().toLowerCase()); nc = PluginMain.NationColor.get(TownyComponent.TU.getTownsMap().get(other.getKey()).getNation().getName().toLowerCase());
} catch (Exception e) { //Too lazy for lots of null-checks and it may throw exceptions anyways } catch (Exception e) { //Too lazy for lots of null-checks and it may throw exceptions anyways
nc = null; nc = null;
} }
@ -103,6 +104,6 @@ public class TownColorCommand extends AdminCommandBase {
} }
public static String getTownNameCased(String name) { public static String getTownNameCased(String name) {
return PluginMain.TU.getTownsMap().get(name.toLowerCase()).getName(); return TownyComponent.TU.getTownsMap().get(name.toLowerCase()).getName();
} }
} }

View file

@ -1,15 +0,0 @@
package buttondevteam.chat.components;
import buttondevteam.lib.architecture.Component;
public class TownyComponent extends Component {
@Override
protected void enable() {
TownyAnnouncer.setup();
}
@Override
protected void disable() {
TownyAnnouncer.setdown();
}
}

View file

@ -1,37 +1,36 @@
package buttondevteam.chat.commands.ucmds.announce; package buttondevteam.chat.components.announce;
import org.bukkit.command.CommandSender; import org.bukkit.command.CommandSender;
import buttondevteam.chat.PluginMain; public class AddCommand extends AnnounceCommandBase {
public class AddCommand extends AnnounceCommandBase { @Override
public String[] GetHelpText(String alias) {
@Override return new String[] {
public String[] GetHelpText(String alias) { "§6---- Add announcement ----",
return new String[] { "This command adds a new announcement",
"§6---- Add announcement ----", "Note: Please avoid using this command, if possible",
"This command adds a new announcement", "Instead, use the command blocks in flatworld to set announcements",
"Note: Please avoid using this command, if possible", "This makes editing announcements easier" };
"Instead, use the command blocks in flatworld to set announcements", }
"This makes editing announcements easier" };
} @Override
public boolean OnCommand(CommandSender sender, String alias,
@Override String[] args) {
public boolean OnCommand(CommandSender sender, String alias, if (args.length < 1) {
String[] args) { return false;
if (args.length < 1) { }
return false; StringBuilder sb = new StringBuilder();
} for (int i = 0; i < args.length; i++) {
StringBuilder sb = new StringBuilder(); sb.append(args[i]);
for (int i = 0; i < args.length; i++) { if (i != args.length - 1)
sb.append(args[i]); sb.append(" ");
if (i != args.length - 1) }
sb.append(" "); String finalmessage = sb.toString().replace('&', '§');
} AnnouncerComponent component = (AnnouncerComponent) getComponent();
String finalmessage = sb.toString().replace('&', '§'); component.AnnounceMessages().get().add(finalmessage);
PluginMain.AnnounceMessages.add(finalmessage); sender.sendMessage("§bAnnouncement added. - Plase avoid using this command if possible, see /u announce add without args.§r");
sender.sendMessage("§bAnnouncement added. - Plase avoid using this command if possible, see /u announce add without args.§r"); return true;
return true; }
}
}
}

View file

@ -1,13 +1,13 @@
package buttondevteam.chat.commands.ucmds.announce; package buttondevteam.chat.components.announce;
import buttondevteam.chat.commands.ucmds.UCommandBase; import buttondevteam.chat.commands.ucmds.UCommandBase;
import buttondevteam.lib.chat.CommandClass; import buttondevteam.lib.chat.CommandClass;
import buttondevteam.lib.chat.OptionallyPlayerCommandClass; import buttondevteam.lib.chat.OptionallyPlayerCommandClass;
@CommandClass(modOnly = true) @CommandClass(modOnly = true)
@OptionallyPlayerCommandClass(playerOnly = false) @OptionallyPlayerCommandClass(playerOnly = false)
public abstract class AnnounceCommandBase extends UCommandBase { public abstract class AnnounceCommandBase extends UCommandBase {
public abstract String[] GetHelpText(String alias); public abstract String[] GetHelpText(String alias);
} }

View file

@ -0,0 +1,50 @@
package buttondevteam.chat.components.announce;
import buttondevteam.lib.architecture.Component;
import buttondevteam.lib.architecture.ConfigData;
import org.bukkit.Bukkit;
import java.util.ArrayList;
public class AnnouncerComponent extends Component implements Runnable {
public ConfigData<ArrayList<String>> AnnounceMessages() {
return getConfig().getData("announceMessages", new ArrayList<>(0));
}
public ConfigData<Integer> AnnounceTime() {
return getConfig().getData("announceTime", 15 * 60 * 1000);
}
private static int AnnounceMessageIndex = 0;
@Override
public void run() {
while (isEnabled()) {
try {
Thread.sleep(AnnounceTime().get());
} catch (InterruptedException ex) {
Thread.currentThread().interrupt();
}
if (Bukkit.getOnlinePlayers().size() == 0) continue; //Don't post to Discord if nobody is on
if (AnnounceMessages().get().size() > AnnounceMessageIndex) {
Bukkit.broadcastMessage(AnnounceMessages().get().get(AnnounceMessageIndex));
AnnounceMessageIndex++;
if (AnnounceMessageIndex == AnnounceMessages().get().size())
AnnounceMessageIndex = 0;
}
}
}
@Override
protected void enable() {
registerCommand(new AddCommand());
registerCommand(new EditCommand());
registerCommand(new ListCommand());
registerCommand(new RemoveCommand());
registerCommand(new SetTimeCommand());
new Thread(this).start();
}
@Override
protected void disable() {
}
}

View file

@ -1,45 +1,44 @@
package buttondevteam.chat.commands.ucmds.announce; package buttondevteam.chat.components.announce;
import org.bukkit.command.BlockCommandSender; import org.bukkit.command.BlockCommandSender;
import org.bukkit.command.CommandSender; import org.bukkit.command.CommandSender;
import buttondevteam.chat.PluginMain; public class EditCommand extends AnnounceCommandBase {
public class EditCommand extends AnnounceCommandBase { @Override
public String[] GetHelpText(String alias) {
@Override return new String[] { "§6---- Edit announcement ----",
public String[] GetHelpText(String alias) { "This command can only be used in a command block.",
return new String[] { "§6---- Edit announcement ----", "Usage: /u annonunce edit <index> <text>" };
"This command can only be used in a command block.", }
"Usage: /u annonunce edit <index> <text>" };
} @Override
public boolean OnCommand(CommandSender sender, String alias,
@Override String[] args) {
public boolean OnCommand(CommandSender sender, String alias, if (!(sender instanceof BlockCommandSender)) {
String[] args) { sender.sendMessage("§cError: This command can only be used from a command block. You can use add and remove, though it's not recommended.");
if (!(sender instanceof BlockCommandSender)) { return true;
sender.sendMessage("§cError: This command can only be used from a command block. You can use add and remove, though it's not recommended."); }
return true; if (args.length < 1) {
} return false;
if (args.length < 1) { }
return false; StringBuilder sb1 = new StringBuilder();
} for (int i1 = 1; i1 < args.length; i1++) {
StringBuilder sb1 = new StringBuilder(); sb1.append(args[i1]);
for (int i1 = 1; i1 < args.length; i1++) { if (i1 != args.length - 1)
sb1.append(args[i1]); sb1.append(" ");
if (i1 != args.length - 1) }
sb1.append(" "); String finalmessage1 = sb1.toString().replace('&', '§');
} int index = Integer.parseInt(args[0]);
String finalmessage1 = sb1.toString().replace('&', '§'); if (index > 100)
int index = Integer.parseInt(args[0]); return false;
if (index > 100) AnnouncerComponent component = (AnnouncerComponent) getComponent();
return false; while (component.AnnounceMessages().get().size() <= index)
while (PluginMain.AnnounceMessages.size() <= index) component.AnnounceMessages().get().add("");
PluginMain.AnnounceMessages.add(""); component.AnnounceMessages().get().set(Integer.parseInt(args[0]),
PluginMain.AnnounceMessages.set(Integer.parseInt(args[0]), finalmessage1);
finalmessage1); sender.sendMessage("Announcement edited.");
sender.sendMessage("Announcement edited."); return true;
return true; }
}
}
}

View file

@ -1,28 +1,27 @@
package buttondevteam.chat.commands.ucmds.announce; package buttondevteam.chat.components.announce;
import org.bukkit.command.CommandSender; import org.bukkit.command.CommandSender;
import buttondevteam.chat.PluginMain; public class ListCommand extends AnnounceCommandBase {
public class ListCommand extends AnnounceCommandBase { @Override
public String[] GetHelpText(String alias) {
@Override return new String[] { "§6---- List announcements ----",
public String[] GetHelpText(String alias) { "This command lists the announcements and the time between them" };
return new String[] { "§6---- List announcements ----", }
"This command lists the announcements and the time between them" };
} @Override
public boolean OnCommand(CommandSender sender, String alias,
@Override String[] args) {
public boolean OnCommand(CommandSender sender, String alias, sender.sendMessage("§bList of announce messages:§r");
String[] args) { sender.sendMessage("§bFormat: [index] message§r");
sender.sendMessage("§bList of announce messages:§r"); int i = 0;
sender.sendMessage("§bFormat: [index] message§r"); AnnouncerComponent component = (AnnouncerComponent) getComponent();
int i = 0; for (String message : component.AnnounceMessages().get())
for (String message : PluginMain.AnnounceMessages) sender.sendMessage("[" + i++ + "] " + message);
sender.sendMessage("[" + i++ + "] " + message); sender.sendMessage("§bCurrent wait time between announcements: "
sender.sendMessage("§bCurrent wait time between announcements: " + component.AnnounceTime().get() / 60 / 1000 + " minute(s)§r");
+ PluginMain.AnnounceTime / 60 / 1000 + " minute(s)§r"); return true;
return true; }
}
}
}

View file

@ -1,30 +1,28 @@
package buttondevteam.chat.commands.ucmds.announce; package buttondevteam.chat.components.announce;
import org.bukkit.command.CommandSender; import org.bukkit.command.CommandSender;
import buttondevteam.chat.PluginMain; public class RemoveCommand extends AnnounceCommandBase {
public class RemoveCommand extends AnnounceCommandBase { @Override
public String[] GetHelpText(String alias) {
@Override return new String[] {
public String[] GetHelpText(String alias) { "§6---- Remove announcement ----",
return new String[] { "This command removes an announcement",
"§6---- Remove announcement ----", "Note: Please avoid using this command, if possible",
"This command removes an announcement", "Instead, use the command blocks in flatworld to set announcements",
"Note: Please avoid using this command, if possible", "This makes editing announcements easier" };
"Instead, use the command blocks in flatworld to set announcements", }
"This makes editing announcements easier" };
} @Override
public boolean OnCommand(CommandSender sender, String alias,
@Override String[] args) {
public boolean OnCommand(CommandSender sender, String alias, if (args.length < 1) {
String[] args) { sender.sendMessage("§cUsage: /u announce remove <index>");
if (args.length < 1) { return true;
sender.sendMessage("§cUsage: /u announce remove <index>"); }
return true; ((AnnouncerComponent) getComponent()).AnnounceMessages().get().remove(Integer.parseInt(args[0]));
} return true;
PluginMain.AnnounceMessages.remove(Integer.parseInt(args[0])); }
return true;
} }
}

View file

@ -1,33 +1,31 @@
package buttondevteam.chat.commands.ucmds.announce; package buttondevteam.chat.components.announce;
import org.bukkit.command.CommandSender; import org.bukkit.command.CommandSender;
import buttondevteam.chat.PluginMain; public class SetTimeCommand extends AnnounceCommandBase {
public class SetTimeCommand extends AnnounceCommandBase { @Override
public String[] GetHelpText(String alias) {
@Override return new String[] { "§6---- Set time ----",
public String[] GetHelpText(String alias) { "This command sets the time between the announcements",
return new String[] { "§6---- Set time ----", "Usage: /u anonunce settime <minutes>", "Default: 15" };
"This command sets the time between the announcements", }
"Usage: /u anonunce settime <minutes>", "Default: 15" };
} @Override
public boolean OnCommand(CommandSender sender, String alias,
@Override String[] args) {
public boolean OnCommand(CommandSender sender, String alias, if (args.length < 1) {
String[] args) { return false;
if (args.length < 1) { }
return false; try {
} ((AnnouncerComponent) getComponent()).AnnounceTime().set(Integer.parseInt(args[0]) * 60 * 1000);
try { } catch (Exception e) {
PluginMain.AnnounceTime = Integer.parseInt(args[0]) * 60 * 1000; sender.sendMessage("§cMinutes argument must be a number. Got: "
} catch (Exception e) { + args[0]);
sender.sendMessage("§cMinutes argument must be a number. Got: " return true;
+ args[0]); }
return true; sender.sendMessage("Time set between announce messages");
} return true;
sender.sendMessage("Time set between announce messages"); }
return true;
} }
}

View file

@ -1,99 +1,97 @@
package buttondevteam.chat.commands.ucmds; package buttondevteam.chat.components.flair;
import java.util.Timer; import buttondevteam.chat.ChatPlayer;
import buttondevteam.chat.PlayerJoinTimerTask;
import org.bukkit.command.CommandSender; import buttondevteam.chat.commands.ucmds.UCommandBase;
import org.bukkit.entity.Player; import buttondevteam.lib.TBMCCoreAPI;
import buttondevteam.lib.chat.CommandClass;
import buttondevteam.chat.ChatPlayer; import buttondevteam.lib.chat.OptionallyPlayerCommandClass;
import buttondevteam.chat.FlairStates; import buttondevteam.lib.player.TBMCPlayer;
import buttondevteam.chat.PlayerJoinTimerTask; import org.bukkit.command.CommandSender;
import buttondevteam.chat.PluginMain; import org.bukkit.entity.Player;
import buttondevteam.lib.TBMCCoreAPI;
import buttondevteam.lib.chat.CommandClass; import java.util.Timer;
import buttondevteam.lib.chat.OptionallyPlayerCommandClass;
import buttondevteam.lib.player.TBMCPlayer; @CommandClass(modOnly = false)
@OptionallyPlayerCommandClass(playerOnly = true)
@CommandClass(modOnly = false) public class AcceptCommand extends UCommandBase {
@OptionallyPlayerCommandClass(playerOnly = true)
public class AcceptCommand extends UCommandBase { @Override
public String[] GetHelpText(String alias) {
@Override return new String[] { "§6---- Accept flair ----", //
public String[] GetHelpText(String alias) { "Accepts a flair from Reddit", //
return new String[] { "§6---- Accept flair ----", // "Use /u accept <username> if you commented from multiple accounts" //
"Accepts a flair from Reddit", // };
"Use /u accept <username> if you commented from multiple accounts" // }
};
} @Override
public boolean OnCommand(CommandSender sender, String alias, String[] args) {
@Override final Player player = (Player) sender;
public boolean OnCommand(CommandSender sender, String alias, String[] args) { ChatPlayer p = TBMCPlayer.getPlayer(player.getUniqueId(), ChatPlayer.class);
final Player player = (Player) sender; if (args.length < 1 && p.UserNames().size() > 1) {
ChatPlayer p = TBMCPlayer.getPlayer(player.getUniqueId(), ChatPlayer.class); player.sendMessage("§9Multiple users commented your name. §bPlease pick one using /u accept <username>");
if (args.length < 1 && p.UserNames().size() > 1) { StringBuilder sb = new StringBuilder();
player.sendMessage("§9Multiple users commented your name. §bPlease pick one using /u accept <username>"); sb.append("§6Usernames:");
StringBuilder sb = new StringBuilder(); for (String username : p.UserNames())
sb.append("§6Usernames:"); sb.append(" ").append(username);
for (String username : p.UserNames()) player.sendMessage(sb.toString());
sb.append(" ").append(username); return true;
player.sendMessage(sb.toString()); }
return true; if (p.FlairState().get().equals(FlairStates.NoComment) || p.UserNames().size() == 0) {
} player.sendMessage("§cError: You need to write your username to the reddit thread at /r/ChromaGamers§r");
if (p.FlairState().get().equals(FlairStates.NoComment) || p.UserNames().size() == 0) { return true;
player.sendMessage("§cError: You need to write your username to the reddit thread at /r/ChromaGamers§r"); }
return true; if (args.length > 0 && !p.UserNames().contains(args[0])) {
} player.sendMessage("§cError: Unknown name: " + args[0] + "§r");
if (args.length > 0 && !p.UserNames().contains(args[0])) { return true;
player.sendMessage("§cError: Unknown name: " + args[0] + "§r"); }
return true; if (p.Working) {
} player.sendMessage("§cError: Something is already in progress.§r");
if (p.Working) { return true;
player.sendMessage("§cError: Something is already in progress.§r"); }
return true;
} if ((args.length > 0 ? args[0] : p.UserNames().get(0)).equals(p.UserName().get())) {
player.sendMessage("§cYou already have this user's flair.§r");
if ((args.length > 0 ? args[0] : p.UserNames().get(0)).equals(p.UserName().get())) { return true;
player.sendMessage("§cYou already have this user's flair.§r"); }
return true; if (args.length > 0)
} p.UserName().set(args[0]);
if (args.length > 0) else
p.UserName().set(args[0]); p.UserName().set(p.UserNames().get(0));
else
p.UserName().set(p.UserNames().get(0)); player.sendMessage("§bObtaining flair...");
p.Working = true;
player.sendMessage("§bObtaining flair..."); Timer timer = new Timer();
p.Working = true; PlayerJoinTimerTask tt = new PlayerJoinTimerTask() {
Timer timer = new Timer(); @Override
PlayerJoinTimerTask tt = new PlayerJoinTimerTask() { public void run() {
@Override try {
public void run() { FlairComponent.DownloadFlair(mp);
try { } catch (Exception e) {
PluginMain.Instance.DownloadFlair(mp); TBMCCoreAPI.SendException(
} catch (Exception e) { "An error occured while downloading flair for " + player.getCustomName() + "!", e);
TBMCCoreAPI.SendException( player.sendMessage(
"An error occured while downloading flair for " + player.getCustomName() + "!", e); "Sorry, but an error occured while trying to get your flair. Please contact a mod.");
player.sendMessage( mp.Working = false;
"Sorry, but an error occured while trying to get your flair. Please contact a mod."); return;
mp.Working = false; }
return;
} if (mp.FlairState().get().equals(FlairStates.Commented)) {
player.sendMessage(
if (mp.FlairState().get().equals(FlairStates.Commented)) { "Sorry, but your flair isn't recorded. Please ask an admin to set it for you. Also, prepare a comment on /r/thebutton, if possible.");
player.sendMessage( mp.Working = false;
"Sorry, but your flair isn't recorded. Please ask an admin to set it for you. Also, prepare a comment on /r/thebutton, if possible."); return;
mp.Working = false; }
return; String flair = mp.GetFormattedFlair();
} mp.FlairState().set(FlairStates.Accepted);
String flair = mp.GetFormattedFlair(); FlairComponent.ConfirmUserMessage(mp);
mp.FlairState().set(FlairStates.Accepted); player.sendMessage("§bYour flair has been set:§r " + flair);
PluginMain.ConfirmUserMessage(mp); mp.Working = false;
player.sendMessage("§bYour flair has been set:§r " + flair); }
mp.Working = false; };
} tt.mp = p;
}; timer.schedule(tt, 20);
tt.mp = p; return true;
timer.schedule(tt, 20); }
return true;
} }
}

View file

@ -0,0 +1,185 @@
package buttondevteam.chat.components.flair;
import buttondevteam.chat.ChatPlayer;
import buttondevteam.lib.TBMCCoreAPI;
import buttondevteam.lib.architecture.Component;
import buttondevteam.lib.architecture.ConfigData;
import buttondevteam.lib.player.TBMCPlayerBase;
import com.google.gson.JsonArray;
import com.google.gson.JsonObject;
import com.google.gson.JsonParser;
import org.bukkit.Bukkit;
import org.bukkit.entity.Player;
import org.htmlcleaner.HtmlCleaner;
import org.htmlcleaner.TagNode;
import java.io.IOException;
import java.io.InputStream;
import java.net.URL;
import java.net.URLConnection;
import java.net.UnknownHostException;
import java.text.SimpleDateFormat;
import java.util.*;
public class FlairComponent extends Component {
private ConfigData<String> FlairThreadURL() {
return getConfig().getData("flairThreadURL", "https://www.reddit.com/r/Chromagamers/comments/51ys94/flair_thread_for_the_mc_server/");
}
/**
* <p>
* This variable is used as a cache for flair state checking when reading the flair thread.
* </p>
* <p>
* It's used because normally it has to load all associated player files every time to read the flair state
* </p>
*/
private Set<String> PlayersWithFlairs = new HashSet<>();
@Override
protected void enable() {
registerCommand(new AcceptCommand());
registerCommand(new IgnoreCommand());
registerCommand(new SetFlairCommand());
new Thread(this::FlairGetterThreadMethod).start();
}
@Override
protected void disable() {
}
private void FlairGetterThreadMethod() {
int errorcount = 0;
while (isEnabled()) {
try {
String body = TBMCCoreAPI.DownloadString(FlairThreadURL().get() + ".json?limit=1000");
JsonArray json = new JsonParser().parse(body).getAsJsonArray().get(1).getAsJsonObject().get("data")
.getAsJsonObject().get("children").getAsJsonArray();
for (Object obj : json) {
JsonObject item = (JsonObject) obj;
String author = item.get("data").getAsJsonObject().get("author").getAsString();
String ign = item.get("data").getAsJsonObject().get("body").getAsString();
int start = ign.indexOf("IGN:") + "IGN:".length();
if (start == -1 + "IGN:".length())
continue;
int end = ign.indexOf(' ', start);
if (end == -1 || end == start)
end = ign.indexOf('\n', start);
if (end == -1 || end == start)
ign = ign.substring(start);
else
ign = ign.substring(start, end);
ign = ign.trim();
if (PlayersWithFlairs.contains(ign))
continue;
try (ChatPlayer mp = TBMCPlayerBase.getFromName(ign, ChatPlayer.class)) { // Loads player file
if (mp == null)
continue;
/*
* if (!JoinedBefore(mp, 2015, 6, 5)) continue;
*/
if (!mp.UserNames().contains(author))
mp.UserNames().add(author);
if (mp.FlairState().get().equals(FlairStates.NoComment)) {
mp.FlairState().set(FlairStates.Commented);
ConfirmUserMessage(mp);
}
PlayersWithFlairs.add(ign); // Don't redownload even if flair isn't accepted
}
}
} catch (Exception e) {
errorcount++;
if (errorcount >= 10) {
errorcount = 0;
if (!e.getMessage().contains("Server returned HTTP response code")
&& !(e instanceof UnknownHostException))
TBMCCoreAPI.SendException("Error while getting flairs from Reddit!", e);
}
}
try {
Thread.sleep(10000);
} catch (InterruptedException ex) {
Thread.currentThread().interrupt();
}
}
}
public static void DownloadFlair(ChatPlayer mp) throws IOException {
String[] flairdata = TBMCCoreAPI
.DownloadString("http://karmadecay.com/thebutton-data.php?users=" + mp.UserName().get())
.replace("\"", "").split(":");
String flair;
if (flairdata.length > 1)
flair = flairdata[1];
else
flair = "";
String flairclass;
if (flairdata.length > 2)
flairclass = flairdata[2];
else
flairclass = "unknown";
SetFlair(mp, flair, flairclass, mp.UserName().get());
}
private static void SetFlair(ChatPlayer p, String text, String flairclass, String username) {
p.UserName().set(username);
p.FlairState().set(FlairStates.Recognised);
switch (flairclass) {
case "cheater":
p.SetFlair(Short.parseShort(text), true);
return;
case "unknown":
try {
if (CheckForJoinDate(p)) {
if (text.equals("-1")) // If true, only non-presser/can't press; if false, any flair (but we can still detect can't press)
p.SetFlair(ChatPlayer.FlairTimeNonPresser);
else
p.SetFlair(ChatPlayer.FlairTimeNone); // Flair unknown
} else {
p.SetFlair(ChatPlayer.FlairTimeCantPress);
}
} catch (Exception e) {
p.FlairState().set(FlairStates.Commented); // Flair unknown
p.SetFlair(ChatPlayer.FlairTimeNone);
TBMCCoreAPI.SendException("Error while checking join date for player " + p.PlayerName() + "!", e);
}
return;
default:
break;
}
p.SetFlair(Short.parseShort(text));
}
private static boolean CheckForJoinDate(ChatPlayer mp) throws Exception {
return JoinedBefore(mp, 2015, 4, 1);
}
private static boolean JoinedBefore(ChatPlayer mp, int year, int month, int day) throws Exception {
URL url = new URL("https://www.reddit.com/u/" + mp.UserName());
URLConnection con = url.openConnection();
con.setRequestProperty("User-Agent", "TheButtonAutoFlair");
InputStream in = con.getInputStream();
HtmlCleaner cleaner = new HtmlCleaner();
TagNode node = cleaner.clean(in);
node = node.getElementsByAttValue("class", "age", true, true)[0];
node = node.getElementsByName("time", false)[0];
String joindate = node.getAttributeByName("datetime");
SimpleDateFormat parserSDF = new SimpleDateFormat("yyyy-MM-dd");
joindate = joindate.split("T")[0];
Date date = parserSDF.parse(joindate);
return date.before(new Calendar.Builder().setTimeZone(TimeZone.getTimeZone("UTC")).setDate(year, month, day)
.build().getTime());
}
public static void ConfirmUserMessage(ChatPlayer mp) {
Player p = Bukkit.getPlayer(mp.getUUID());
if (mp.FlairState().get().equals(FlairStates.Commented) && p != null)
if (mp.UserNames().size() > 1)
p.sendMessage(
"§9Multiple Reddit users commented your name. You can select with /u accept.§r §6Type /u accept or /u ignore§r");
else
p.sendMessage("§9A Reddit user commented your name. Is that you?§r §6Type /u accept or /u ignore§r");
}
}

View file

@ -1,5 +1,5 @@
package buttondevteam.chat; package buttondevteam.chat.components.flair;
public enum FlairStates { public enum FlairStates {
Accepted, Ignored, Recognised, Commented, NoComment Accepted, Ignored, Recognised, Commented, NoComment
} }

View file

@ -1,44 +1,43 @@
package buttondevteam.chat.commands.ucmds; package buttondevteam.chat.components.flair;
import org.bukkit.command.CommandSender; import buttondevteam.chat.ChatPlayer;
import org.bukkit.entity.Player; import buttondevteam.chat.commands.ucmds.UCommandBase;
import buttondevteam.lib.chat.CommandClass;
import buttondevteam.chat.ChatPlayer; import buttondevteam.lib.chat.OptionallyPlayerCommandClass;
import buttondevteam.chat.FlairStates; import buttondevteam.lib.player.TBMCPlayer;
import buttondevteam.lib.chat.CommandClass; import org.bukkit.command.CommandSender;
import buttondevteam.lib.chat.OptionallyPlayerCommandClass; import org.bukkit.entity.Player;
import buttondevteam.lib.player.TBMCPlayer;
@CommandClass(modOnly = false)
@CommandClass(modOnly = false) @OptionallyPlayerCommandClass(playerOnly = true)
@OptionallyPlayerCommandClass(playerOnly = true) public final class IgnoreCommand extends UCommandBase {
public final class IgnoreCommand extends UCommandBase {
@Override
@Override public String[] GetHelpText(String alias) {
public String[] GetHelpText(String alias) { return new String[] { "§6---- Ignore flair ----",
return new String[] { "§6---- Ignore flair ----", "Stop the \"write your name in the thread\" message from showing up" };
"Stop the \"write your name in the thread\" message from showing up" }; }
}
@Override
@Override public boolean OnCommand(CommandSender sender, String alias, String[] args) {
public boolean OnCommand(CommandSender sender, String alias, String[] args) { final Player player = (Player) sender;
final Player player = (Player) sender; ChatPlayer p = TBMCPlayer.getPlayer(player.getUniqueId(), ChatPlayer.class);
ChatPlayer p = TBMCPlayer.getPlayer(player.getUniqueId(), ChatPlayer.class); if (p.FlairState().get().equals(FlairStates.Accepted)) {
if (p.FlairState().get().equals(FlairStates.Accepted)) { player.sendMessage("§cYou can only ignore the \"write your name in the thread\" message.");
player.sendMessage("§cYou can only ignore the \"write your name in the thread\" message."); return true;
return true; }
} if (p.FlairState().get().equals(FlairStates.Commented)) {
if (p.FlairState().get().equals(FlairStates.Commented)) { player.sendMessage("Sorry, but your flair isn't recorded. Please ask a mod to set it for you.");
player.sendMessage("Sorry, but your flair isn't recorded. Please ask a mod to set it for you."); return true;
return true; }
} if (!p.FlairState().get().equals(FlairStates.Ignored)) {
if (!p.FlairState().get().equals(FlairStates.Ignored)) { p.FlairState().set(FlairStates.Ignored);
p.FlairState().set(FlairStates.Ignored); p.SetFlair(ChatPlayer.FlairTimeNone);
p.SetFlair(ChatPlayer.FlairTimeNone); p.UserName().set("");
p.UserName().set(""); player.sendMessage("§bYou have ignored the message.§r");
player.sendMessage("§bYou have ignored the message.§r"); } else
} else player.sendMessage("§cYou already ignored the message.§r");
player.sendMessage("§cYou already ignored the message.§r"); return true;
return true; }
}
}
}

View file

@ -1,71 +1,70 @@
package buttondevteam.chat.commands.ucmds.admin; package buttondevteam.chat.components.flair;
import org.bukkit.Bukkit; import buttondevteam.chat.ChatPlayer;
import org.bukkit.command.CommandSender; import buttondevteam.chat.commands.ucmds.admin.AdminCommandBase;
import org.bukkit.entity.Player; import buttondevteam.lib.player.TBMCPlayerBase;
import org.bukkit.Bukkit;
import buttondevteam.chat.ChatPlayer; import org.bukkit.command.CommandSender;
import buttondevteam.chat.FlairStates; import org.bukkit.entity.Player;
import buttondevteam.lib.player.TBMCPlayerBase;
public class SetFlairCommand extends AdminCommandBase {
public class SetFlairCommand extends AdminCommandBase {
@Override
@Override public String[] GetHelpText(String alias) {
public String[] GetHelpText(String alias) { return new String[] { "§6---- Set flair -----", "Set a flair for a player",
return new String[] { "§6---- Set flair -----", "Set a flair for a player", "Usage: /u admin setflair <player> <flairtime (or non-presser, cant-press, none)> <cheater(true/false)> [username]",
"Usage: /u admin setflair <player> <flairtime (or non-presser, cant-press, none)> <cheater(true/false)> [username]", "Example 1: /u admin setflair NorbiPeti 19 false NorbiPeti --> orange (19s)",
"Example 1: /u admin setflair NorbiPeti 19 false NorbiPeti --> orange (19s)", "Example 2: /u admin setflair iie 0 true asde --> purple (0s)" };
"Example 2: /u admin setflair iie 0 true asde --> purple (0s)" }; }
}
@Override
@Override public boolean OnCommand(CommandSender sender, String alias, String[] args) {
public boolean OnCommand(CommandSender sender, String alias, String[] args) { if (args.length < 3) {
if (args.length < 3) { return false;
return false; }
} Player p = Bukkit.getPlayer(args[0]);
Player p = Bukkit.getPlayer(args[0]); if (p == null) {
if (p == null) { sender.sendMessage("§cPlayer not found.&r");
sender.sendMessage("§cPlayer not found.&r"); return true;
return true; }
} short flairtime = 0x00;
short flairtime = 0x00; if (args[1].equalsIgnoreCase("non-presser"))
if (args[1].equalsIgnoreCase("non-presser")) flairtime = ChatPlayer.FlairTimeNonPresser;
flairtime = ChatPlayer.FlairTimeNonPresser; else if (args[1].equalsIgnoreCase("cant-press"))
else if (args[1].equalsIgnoreCase("cant-press")) flairtime = ChatPlayer.FlairTimeCantPress;
flairtime = ChatPlayer.FlairTimeCantPress; else if (args[1].equalsIgnoreCase("none"))
else if (args[1].equalsIgnoreCase("none")) flairtime = ChatPlayer.FlairTimeNone;
flairtime = ChatPlayer.FlairTimeNone; else {
else { try {
try { flairtime = Short.parseShort(args[1]);
flairtime = Short.parseShort(args[1]); } catch (Exception e) {
} catch (Exception e) { sender.sendMessage(
sender.sendMessage( "§cFlairtime must be a number, \"non-presser\", \"cant-press\" or \"none\". Run without args to see usage.");
"§cFlairtime must be a number, \"non-presser\", \"cant-press\" or \"none\". Run without args to see usage."); return true;
return true; }
} }
} boolean cheater;
boolean cheater = false; if (args[2].equalsIgnoreCase("true"))
if (args[2].equalsIgnoreCase("true")) cheater = true;
cheater = true; else if (args[2].equalsIgnoreCase("false"))
else if (args[2].equalsIgnoreCase("false")) cheater = false;
cheater = false; else {
else { sender.sendMessage("§cUnknown value for cheater parameter.");
sender.sendMessage("§cUnknown value for cheater parameter. Run without args to see usage."); return false;
return true; }
} ChatPlayer mp = TBMCPlayerBase.getPlayer(p.getUniqueId(), ChatPlayer.class);
ChatPlayer mp = TBMCPlayerBase.getPlayer(p.getUniqueId(), ChatPlayer.class); mp.SetFlair(flairtime, cheater);
mp.SetFlair(flairtime, cheater); mp.FlairState().set(FlairStates.Accepted);
mp.FlairState().set(FlairStates.Accepted); if (args.length < 4)
if (args.length < 4) mp.UserName().set("");
mp.UserName().set(""); else {
else { mp.UserName().set(args[3]);
mp.UserName().set(args[3]); if (!mp.UserNames().contains(args[3]))
if (!mp.UserNames().contains(args[3])) mp.UserNames().add(args[3]);
mp.UserNames().add(args[3]); }
} sender.sendMessage(
sender.sendMessage( "§bThe flair has been set. Player: " + mp.PlayerName() + " Flair: " + mp.GetFormattedFlair() + "§r");
"§bThe flair has been set. Player: " + mp.PlayerName() + " Flair: " + mp.GetFormattedFlair() + "§r"); return true;
return true; }
}
}
}

View file

@ -1,4 +1,4 @@
package buttondevteam.chat.components; package buttondevteam.chat.components.towny;
import buttondevteam.chat.ChatProcessing; import buttondevteam.chat.ChatProcessing;
import buttondevteam.chat.PluginMain; import buttondevteam.chat.PluginMain;
@ -23,12 +23,12 @@ public class TownyAnnouncer {
switch (String.valueOf(m.group(1))) { //valueOf: Handles null switch (String.valueOf(m.group(1))) { //valueOf: Handles null
case "Town": case "Town":
TBMCChatAPI.SendSystemMessage(PluginMain.TownChat, TBMCChatAPI.SendSystemMessage(PluginMain.TownChat,
new Channel.RecipientTestResult(PluginMain.getTownNationIndex(groupID, false), groupID), new Channel.RecipientTestResult(TownyComponent.getTownNationIndex(groupID, false), groupID),
logRecord.getMessage(), ChatProcessing.MCORIGIN); logRecord.getMessage(), ChatProcessing.MCORIGIN);
break; break;
case "Nation": case "Nation":
TBMCChatAPI.SendSystemMessage(PluginMain.NationChat, TBMCChatAPI.SendSystemMessage(PluginMain.NationChat,
new Channel.RecipientTestResult(PluginMain.getTownNationIndex(groupID, true), groupID), new Channel.RecipientTestResult(TownyComponent.getTownNationIndex(groupID, true), groupID),
logRecord.getMessage(), ChatProcessing.MCORIGIN); logRecord.getMessage(), ChatProcessing.MCORIGIN);
break; break;
case "Global": case "Global":

View file

@ -0,0 +1,94 @@
package buttondevteam.chat.components.towny;
import buttondevteam.chat.PluginMain;
import buttondevteam.component.channel.Channel;
import buttondevteam.lib.architecture.Component;
import buttondevteam.lib.chat.Color;
import buttondevteam.lib.chat.TBMCChatAPI;
import com.palmergames.bukkit.towny.Towny;
import com.palmergames.bukkit.towny.exceptions.NotRegisteredException;
import com.palmergames.bukkit.towny.object.Nation;
import com.palmergames.bukkit.towny.object.Resident;
import com.palmergames.bukkit.towny.object.Town;
import com.palmergames.bukkit.towny.object.TownyUniverse;
import lombok.val;
import org.bukkit.Bukkit;
import org.bukkit.command.CommandSender;
import org.bukkit.entity.Player;
import java.util.ArrayList;
import java.util.stream.Collectors;
public class TownyComponent extends Component {
public static TownyUniverse TU;
private static ArrayList<String> Towns;
private static ArrayList<String> Nations;
@Override
protected void enable() {
TU = ((Towny) Bukkit.getPluginManager().getPlugin("Towny")).getTownyUniverse();
Towns = TU.getTownsMap().values().stream().map(Town::getName).collect(Collectors.toCollection(ArrayList::new)); // Creates a snapshot of towns, new towns will be added when needed
Nations = TU.getNationsMap().values().stream().map(Nation::getName).collect(Collectors.toCollection(ArrayList::new)); // Same here but with nations
TBMCChatAPI.RegisterChatChannel(
PluginMain.TownChat = new Channel("§3TC§f", Color.DarkAqua, "tc", s -> checkTownNationChat(s, false)));
TBMCChatAPI.RegisterChatChannel(
PluginMain.NationChat = new Channel("§6NC§f", Color.Gold, "nc", s -> checkTownNationChat(s, true)));
TownyAnnouncer.setup();
}
@Override
protected void disable() {
TownyAnnouncer.setdown();
}
/**
* Return the error message for the message sender if they can't send it and the score
*/
private static Channel.RecipientTestResult checkTownNationChat(CommandSender sender, boolean nationchat) {
if (!(sender instanceof Player))
return new Channel.RecipientTestResult("§cYou are not a player!");
Resident resident = TU.getResidentMap().get(sender.getName().toLowerCase());
Channel.RecipientTestResult result = checkTownNationChatInternal(sender, nationchat, resident);
if (result.errormessage != null && resident != null && resident.getModes().contains("spy")) // Only use spy if they wouldn't see it
result = new Channel.RecipientTestResult(1000, "allspies"); // There won't be more than a thousand towns/nations probably
return result;
}
private static Channel.RecipientTestResult checkTownNationChatInternal(CommandSender sender, boolean nationchat,
Resident resident) {
try {
/*
* p.sendMessage(String.format("[SPY-%s] - %s: %s", channel.DisplayName, ((Player) sender).getDisplayName(), message));
*/
Town town = null;
if (resident != null && resident.hasTown())
town = resident.getTown();
if (town == null)
return new Channel.RecipientTestResult("You aren't in a town.");
Nation nation = null;
int index;
if (nationchat) {
if (town.hasNation())
nation = town.getNation();
if (nation == null)
return new Channel.RecipientTestResult("Your town isn't in a nation.");
index = getTownNationIndex(nation.getName(), true);
} else
index = getTownNationIndex(town.getName(), false);
return new Channel.RecipientTestResult(index, nationchat ? nation.getName() : town.getName());
} catch (NotRegisteredException e) {
return new Channel.RecipientTestResult("You (probably) aren't knwon by Towny! (Not in a town)");
}
}
public static int getTownNationIndex(String name, boolean nation) {
val list = nation ? Nations : Towns;
int index = list.indexOf(name);
if (index < 0) {
list.add(name);
index = list.size() - 1;
}
return index;
}
}

View file

@ -1,144 +1,149 @@
package buttondevteam.chat.listener; package buttondevteam.chat.listener;
import buttondevteam.chat.ChatPlayer; import buttondevteam.chat.ChatPlayer;
import buttondevteam.chat.FlairStates; import buttondevteam.chat.PlayerJoinTimerTask;
import buttondevteam.chat.PlayerJoinTimerTask; import buttondevteam.chat.PluginMain;
import buttondevteam.chat.PluginMain; import buttondevteam.chat.commands.UnlolCommand;
import buttondevteam.chat.commands.UnlolCommand; import buttondevteam.chat.commands.ucmds.HistoryCommand;
import buttondevteam.chat.commands.ucmds.HistoryCommand; import buttondevteam.chat.components.flair.FlairComponent;
import buttondevteam.lib.chat.Color; import buttondevteam.chat.components.flair.FlairStates;
import buttondevteam.lib.player.TBMCPlayerJoinEvent; import buttondevteam.chat.components.towny.TownyComponent;
import buttondevteam.lib.player.TBMCPlayerLoadEvent; import buttondevteam.core.ComponentManager;
import buttondevteam.lib.player.TBMCPlayerSaveEvent; import buttondevteam.lib.chat.Color;
import com.earth2me.essentials.User; import buttondevteam.lib.player.TBMCPlayerJoinEvent;
import com.palmergames.bukkit.towny.exceptions.NotRegisteredException; import buttondevteam.lib.player.TBMCPlayerLoadEvent;
import lombok.val; import buttondevteam.lib.player.TBMCPlayerSaveEvent;
import org.bukkit.Bukkit; import com.earth2me.essentials.User;
import org.bukkit.ChatColor; import com.palmergames.bukkit.towny.exceptions.NotRegisteredException;
import org.bukkit.GameMode; import lombok.val;
import org.bukkit.entity.Player; import org.bukkit.Bukkit;
import org.bukkit.event.EventHandler; import org.bukkit.ChatColor;
import org.bukkit.event.Listener; import org.bukkit.GameMode;
import org.bukkit.event.player.PlayerQuitEvent; import org.bukkit.entity.Player;
import org.bukkit.event.EventHandler;
import java.util.Arrays; import org.bukkit.event.Listener;
import java.util.Optional; import org.bukkit.event.player.PlayerQuitEvent;
import java.util.Timer;
import java.util.concurrent.atomic.AtomicInteger; import java.util.Arrays;
import java.util.function.BiFunction; import java.util.Optional;
import java.util.Timer;
public class PlayerJoinLeaveListener implements Listener { import java.util.concurrent.atomic.AtomicInteger;
import java.util.function.BiFunction;
@EventHandler
public void onPlayerLoad(TBMCPlayerLoadEvent e) { public class PlayerJoinLeaveListener implements Listener {
ChatPlayer cp = e.GetPlayer().asPluginPlayer(ChatPlayer.class);
cp.FlairUpdate(); @EventHandler
} public void onPlayerLoad(TBMCPlayerLoadEvent e) {
ChatPlayer cp = e.GetPlayer().asPluginPlayer(ChatPlayer.class);
@EventHandler cp.FlairUpdate();
public void onPlayerTBMCJoin(TBMCPlayerJoinEvent e) { }
ChatPlayer cp = e.GetPlayer().asPluginPlayer(ChatPlayer.class);
Player p = Bukkit.getPlayer(cp.getUUID()); @EventHandler
public void onPlayerTBMCJoin(TBMCPlayerJoinEvent e) {
if (!cp.FlairState().get().equals(FlairStates.NoComment)) { ChatPlayer cp = e.GetPlayer().asPluginPlayer(ChatPlayer.class);
PluginMain.ConfirmUserMessage(cp); Player p = Bukkit.getPlayer(cp.getUUID());
Timer timer = new Timer();
PlayerJoinTimerTask tt = new PlayerJoinTimerTask() { if (ComponentManager.isEnabled(FlairComponent.class)) {
@Override if (!cp.FlairState().get().equals(FlairStates.NoComment)) {
public void run() { FlairComponent.ConfirmUserMessage(cp);
mp.FlairUpdate(); Timer timer = new Timer();
} PlayerJoinTimerTask tt = new PlayerJoinTimerTask() {
}; @Override
tt.mp = cp; public void run() {
timer.schedule(tt, 1000); mp.FlairUpdate();
} //TODO: Better Reddit integration (OAuth) }
};
String nwithoutformatting = PluginMain.essentials.getUser(p).getNickname(); tt.mp = cp;
timer.schedule(tt, 1000);
int index; } //TODO: Better Reddit integration (OAuth)
if (nwithoutformatting != null) { }
while ((index = nwithoutformatting.indexOf("§k")) != -1)
nwithoutformatting = nwithoutformatting.replace("§k" + nwithoutformatting.charAt(index + 2), ""); // Support for one random char String nwithoutformatting = PluginMain.essentials.getUser(p).getNickname();
while ((index = nwithoutformatting.indexOf('§')) != -1)
nwithoutformatting = nwithoutformatting.replace("§" + nwithoutformatting.charAt(index + 1), ""); int index;
} else if (nwithoutformatting != null) {
nwithoutformatting = p.getName(); while ((index = nwithoutformatting.indexOf("§k")) != -1)
PlayerListener.nicknames.forcePut(nwithoutformatting.toLowerCase(), p.getUniqueId()); nwithoutformatting = nwithoutformatting.replace("§k" + nwithoutformatting.charAt(index + 2), ""); // Support for one random char
while ((index = nwithoutformatting.indexOf('§')) != -1)
updatePlayerColors(p, cp); //TO!DO: Doesn't have effect - It can help to register the listener nwithoutformatting = nwithoutformatting.replace("§" + nwithoutformatting.charAt(index + 1), "");
} else
if (cp.ChatOnly || p.getGameMode().equals(GameMode.SPECTATOR)) { nwithoutformatting = p.getName();
cp.ChatOnly = false; PlayerListener.nicknames.forcePut(nwithoutformatting.toLowerCase(), p.getUniqueId());
p.setGameMode(GameMode.SURVIVAL);
} updatePlayerColors(p, cp); //TO!DO: Doesn't have effect - It can help to register the listener
HistoryCommand.showHistory(e.getPlayer(), "u history", new String[0], null); if (cp.ChatOnly || p.getGameMode().equals(GameMode.SPECTATOR)) {
} cp.ChatOnly = false;
p.setGameMode(GameMode.SURVIVAL);
@EventHandler }
public void onPlayerSave(TBMCPlayerSaveEvent e) {
} HistoryCommand.showHistory(e.getPlayer(), "u history", new String[0], null);
}
@EventHandler
public void onPlayerLeave(PlayerQuitEvent event) { @EventHandler
PlayerListener.nicknames.inverse().remove(event.getPlayer().getUniqueId()); public void onPlayerSave(TBMCPlayerSaveEvent e) {
UnlolCommand.Lastlol.values().removeIf(lld -> lld.getLolowner().equals(event.getPlayer())); }
}
@EventHandler
private static String getPlayerNickname(Player player, User user, ChatPlayer cp) { public void onPlayerLeave(PlayerQuitEvent event) {
String nickname = user.getNick(true); PlayerListener.nicknames.inverse().remove(event.getPlayer().getUniqueId());
if (nickname.contains("~")) //StartsWith doesn't work because of color codes UnlolCommand.Lastlol.values().removeIf(lld -> lld.getLolowner().equals(event.getPlayer()));
nickname = nickname.replace("~", ""); //It gets stacked otherwise }
String name = ChatColor.stripColor(nickname); //Enforce "town colors" on non-members
val res = PluginMain.TU.getResidentMap().get(player.getName().toLowerCase()); private static String getPlayerNickname(Player player, User user, ChatPlayer cp) {
if (res == null || !res.hasTown()) String nickname = user.getNick(true);
return name; if (nickname.contains("~")) //StartsWith doesn't work because of color codes
try { nickname = nickname.replace("~", ""); //It gets stacked otherwise
Color[] clrs = Optional.ofNullable( String name = ChatColor.stripColor(nickname); //Enforce "town colors" on non-members
PluginMain.TownColors.get(res.getTown().getName().toLowerCase()) val res = TownyComponent.TU.getResidentMap().get(player.getName().toLowerCase());
).orElse(new Color[]{Color.White}); //Use white as default town color if (res == null || !res.hasTown())
StringBuilder ret = new StringBuilder(); return name;
AtomicInteger prevlen = new AtomicInteger(); try {
BiFunction<Color, Integer, String> anyColoredNamePart = (c, len) -> "§" //Len==0 if last part Color[] clrs = Optional.ofNullable(
+ Integer.toHexString(c.ordinal()) // 'Odds' are the last character is chopped off so we make sure to include all chars at the end PluginMain.TownColors.get(res.getTown().getName().toLowerCase())
+ (len == 0 ? name.substring(prevlen.get()) ).orElse(new Color[]{Color.White}); //Use white as default town color
: name.substring(prevlen.get(), prevlen.addAndGet(len))); StringBuilder ret = new StringBuilder();
BiFunction<Integer, Integer, String> coloredNamePart = (len, i) AtomicInteger prevlen = new AtomicInteger();
-> anyColoredNamePart.apply(clrs[i], i + 1 == clrs.length ? 0 : len); BiFunction<Color, Integer, String> anyColoredNamePart = (c, len) -> "§" //Len==0 if last part
final int len = name.length() / (clrs.length + 1); //The above param is needed because this isn't always passed + Integer.toHexString(c.ordinal()) // 'Odds' are the last character is chopped off so we make sure to include all chars at the end
Color nc; + (len == 0 ? name.substring(prevlen.get())
/*if(res.getTown().hasNation() : name.substring(prevlen.get(), prevlen.addAndGet(len)));
&&(nc=PluginMain.NationColor.get(res.getTown().getNation().getName().toLowerCase()))!=null) BiFunction<Integer, Integer, String> coloredNamePart = (len, i)
len = name.length() / (clrs.length+1); -> anyColoredNamePart.apply(clrs[i], i + 1 == clrs.length ? 0 : len);
else final int len = name.length() / (clrs.length + 1); //The above param is needed because this isn't always passed
len = name.length() / clrs.length;*/ Color nc;
val nclar = cp.NameColorLocations().get(); /*if(res.getTown().hasNation()
int[] ncl = nclar == null ? null : nclar.stream().mapToInt(Integer::intValue).toArray(); &&(nc=PluginMain.NationColor.get(res.getTown().getNation().getName().toLowerCase()))!=null)
if (ncl != null && (Arrays.stream(ncl).sum() != name.length() || ncl.length != clrs.length + 1)) //+1: Nation color len = name.length() / (clrs.length+1);
ncl = null; // Reset if name length changed else
//System.out.println("ncl: "+Arrays.toString(ncl)+" - sum: "+Arrays.stream(ncl).sum()+" - name len: "+name.length()); len = name.length() / clrs.length;*/
if (!res.getTown().hasNation() val nclar = cp.NameColorLocations().get();
|| (nc = PluginMain.NationColor.get(res.getTown().getNation().getName().toLowerCase())) == null) int[] ncl = nclar == null ? null : nclar.stream().mapToInt(Integer::intValue).toArray();
nc = Color.White; if (ncl != null && (Arrays.stream(ncl).sum() != name.length() || ncl.length != clrs.length + 1)) //+1: Nation color
ret.append(anyColoredNamePart.apply(nc, ncl == null ? len : ncl[0])); //Make first color the nation color ncl = null; // Reset if name length changed
for (int i = 0; i < clrs.length; i++) //System.out.println("ncl: "+Arrays.toString(ncl)+" - sum: "+Arrays.stream(ncl).sum()+" - name len: "+name.length());
//ret.append(coloredNamePart.apply(ncl == null ? len : (nc==null?ncl[i]:ncl[i+1]), i)); if (!res.getTown().hasNation()
ret.append(coloredNamePart.apply(ncl == null ? len : ncl[i + 1], i)); || (nc = PluginMain.NationColor.get(res.getTown().getNation().getName().toLowerCase())) == null)
return ret.toString(); nc = Color.White;
} catch (NotRegisteredException e) { ret.append(anyColoredNamePart.apply(nc, ncl == null ? len : ncl[0])); //Make first color the nation color
return nickname; for (int i = 0; i < clrs.length; i++)
} //ret.append(coloredNamePart.apply(ncl == null ? len : (nc==null?ncl[i]:ncl[i+1]), i));
} ret.append(coloredNamePart.apply(ncl == null ? len : ncl[i + 1], i));
return ret.toString();
public static void updatePlayerColors(Player player) { //Probably while ingame (/u ncolor) } catch (NotRegisteredException e) {
updatePlayerColors(player, ChatPlayer.getPlayer(player.getUniqueId(), ChatPlayer.class)); return nickname;
} }
}
@SuppressWarnings("WeakerAccess")
public static void updatePlayerColors(Player player, ChatPlayer cp) { //Probably at join - nop, nicknames public static void updatePlayerColors(Player player) { //Probably while ingame (/u ncolor)
User user = PluginMain.essentials.getUser(player); updatePlayerColors(player, ChatPlayer.getPlayer(player.getUniqueId(), ChatPlayer.class));
user.setNickname(getPlayerNickname(player, user, cp)); }
user.setDisplayNick(); //These won't fire the nick change event
cp.FlairUpdate(); //Update in list @SuppressWarnings("WeakerAccess")
} public static void updatePlayerColors(Player player, ChatPlayer cp) { //Probably at join - nop, nicknames
} User user = PluginMain.essentials.getUser(player);
user.setNickname(getPlayerNickname(player, user, cp));
user.setDisplayNick(); //These won't fire the nick change event
cp.FlairUpdate(); //Update in list
}
}

View file

@ -4,9 +4,11 @@ import buttondevteam.chat.ChatPlayer;
import buttondevteam.chat.ChatProcessing; import buttondevteam.chat.ChatProcessing;
import buttondevteam.chat.PluginMain; import buttondevteam.chat.PluginMain;
import buttondevteam.chat.commands.ucmds.HistoryCommand; import buttondevteam.chat.commands.ucmds.HistoryCommand;
import buttondevteam.chat.components.flair.FlairComponent;
import buttondevteam.component.channel.Channel; import buttondevteam.component.channel.Channel;
import buttondevteam.component.channel.ChatChannelRegisterEvent; import buttondevteam.component.channel.ChatChannelRegisterEvent;
import buttondevteam.component.channel.ChatRoom; import buttondevteam.component.channel.ChatRoom;
import buttondevteam.core.ComponentManager;
import buttondevteam.lib.TBMCChatEvent; import buttondevteam.lib.TBMCChatEvent;
import buttondevteam.lib.TBMCCoreAPI; import buttondevteam.lib.TBMCCoreAPI;
import buttondevteam.lib.chat.ChatMessage; import buttondevteam.lib.chat.ChatMessage;
@ -50,15 +52,9 @@ public class PlayerListener implements Listener {
*/ */
public static BiMap<String, UUID> nicknames = HashBiMap.create(); public static BiMap<String, UUID> nicknames = HashBiMap.create();
public static boolean Enable = false;
public static int LoginWarningCountTotal = 5;
public static String NotificationSound; public static String NotificationSound;
public static double NotificationPitch; public static double NotificationPitch;
public static boolean ShowRPTag = false;
public final static String[] LaughStrings = new String[]{"xd", "lel", "lawl", "kek", "lmao", "hue", "hah", "rofl"}; public final static String[] LaughStrings = new String[]{"xd", "lel", "lawl", "kek", "lmao", "hue", "hah", "rofl"};
@EventHandler(priority = EventPriority.HIGHEST) @EventHandler(priority = EventPriority.HIGHEST)
@ -88,7 +84,7 @@ public class PlayerListener implements Listener {
return false; return false;
// ^^ We can only store player or console channels - Directly sending to channels would still work if they had an event // ^^ We can only store player or console channels - Directly sending to channels would still work if they had an event
cmd = sender instanceof ConsoleCommandSender ? message : message.substring(1); cmd = sender instanceof ConsoleCommandSender ? message : message.substring(1);
for (Channel channel : Channel.getChannels()) { for (Channel channel : ((Iterable<Channel>) Channel.getChannels()::iterator)) { //Using Stream.forEach would be too easy
if (checkchid.test(channel, cmd)) { if (checkchid.test(channel, cmd)) {
Channel oldch = mp.channel().get(); Channel oldch = mp.channel().get();
if (oldch instanceof ChatRoom) if (oldch instanceof ChatRoom)
@ -130,7 +126,7 @@ public class PlayerListener implements Listener {
return true; return true;
} }
} else } else
for (Channel channel : Channel.getChannels()) { for (Channel channel : (Iterable<Channel>) Channel.getChannels()::iterator) {
if (checkchid.test(channel, cmd)) { //Apparently method references don't require final variables if (checkchid.test(channel, cmd)) { //Apparently method references don't require final variables
TBMCChatAPI.SendChatMessage(ChatMessage.builder(sender, mp, message.substring(index + 1)).build(), channel); TBMCChatAPI.SendChatMessage(ChatMessage.builder(sender, mp, message.substring(index + 1)).build(), channel);
return true; return true;
@ -173,13 +169,10 @@ public class PlayerListener implements Listener {
public static boolean ActiveF = false; public static boolean ActiveF = false;
public static ChatPlayer FPlayer = null; public static ChatPlayer FPlayer = null;
public static BukkitTask Ftask = null; public static BukkitTask Ftask = null;
public static int AlphaDeaths;
public static ArrayList<CommandSender> Fs = new ArrayList<>(); public static ArrayList<CommandSender> Fs = new ArrayList<>();
@EventHandler @EventHandler
public void onPlayerDeath(PlayerDeathEvent e) { public void onPlayerDeath(PlayerDeathEvent e) {
if (e.getEntity().getName().equals("Alpha_Bacca44"))
AlphaDeaths++;
// MinigamePlayer mgp = Minigames.plugin.pdata.getMinigamePlayer(e.getEntity()); // MinigamePlayer mgp = Minigames.plugin.pdata.getMinigamePlayer(e.getEntity());
if (/* (mgp != null && !mgp.isInMinigame()) && */ new Random().nextBoolean()) { // Don't store Fs for NPCs if (/* (mgp != null && !mgp.isInMinigame()) && */ new Random().nextBoolean()) { // Don't store Fs for NPCs
Runnable tt = () -> { Runnable tt = () -> {
@ -249,9 +242,11 @@ public class PlayerListener implements Listener {
e.addInfo("Minecraft name: " + cp.PlayerName().get()); e.addInfo("Minecraft name: " + cp.PlayerName().get());
if (cp.UserName().get() != null && cp.UserName().get().length() > 0) if (cp.UserName().get() != null && cp.UserName().get().length() > 0)
e.addInfo("Reddit name: " + cp.UserName().get()); e.addInfo("Reddit name: " + cp.UserName().get());
final String flair = cp.GetFormattedFlair(e.getTarget() != InfoTarget.MCCommand); if (ComponentManager.isEnabled(FlairComponent.class)) {
if (flair.length() > 0) final String flair = cp.GetFormattedFlair(e.getTarget() != InfoTarget.MCCommand);
e.addInfo("/r/TheButton flair: " + flair); if (flair.length() > 0)
e.addInfo("/r/TheButton flair: " + flair);
}
e.addInfo(String.format("Respect: %.2f", cp.getF())); e.addInfo(String.format("Respect: %.2f", cp.getF()));
} catch (Exception ex) { } catch (Exception ex) {
TBMCCoreAPI.SendException("Error while providing chat info for player " + e.getPlayer().getFileName(), ex); TBMCCoreAPI.SendException("Error while providing chat info for player " + e.getPlayer().getFileName(), ex);