Command refactoring is in progress...
This commit is contained in:
parent
da87a1293e
commit
bce31de137
4 changed files with 1353 additions and 1750 deletions
71
plugin.yml
71
plugin.yml
|
@ -1,37 +1,34 @@
|
|||
name: TheButtonMCPlugin
|
||||
main: io.github.norbipeti.thebuttonmc.PluginMain
|
||||
version: 4.0
|
||||
commands:
|
||||
u:
|
||||
description: Auto-flair system. Accept or ignore flair.
|
||||
usage: "&cUsage: /u accept|ignore|opme&r"
|
||||
nrp:
|
||||
description: Send message in Out-of-Character.
|
||||
usage: "&cUsage: /nrp <message>&r"
|
||||
ooc:
|
||||
description: Send message in Out-of-Character.
|
||||
usage: "&cUsage: /ooc <message>&r"
|
||||
unlol:
|
||||
description: Unlaugh the last laugh.
|
||||
unlaugh:
|
||||
description: Unlaugh the last laugh.
|
||||
mwiki:
|
||||
description: Search the wiki.
|
||||
usage: "&cUsage: /mwiki [query]&r"
|
||||
dontrunthiscmd:
|
||||
tableflip:
|
||||
description: Flip a table.
|
||||
usage: "&cUsage: /tableflip [message]&r"
|
||||
unflip:
|
||||
description: Unflip a flip.
|
||||
usage: "&cUsage: /unflip [message]&r"
|
||||
chatonly:
|
||||
description: Enable chat only mode. Useful for chat clients.
|
||||
usage: "&cUsage: /chatonly&r"
|
||||
author: NorbiPeti
|
||||
depend: [Essentials, Towny, Minigames, Votifier, Factions, WorldGuard, WorldEdit, ProtocolLib, FastLogin, Vault, AuthMe]
|
||||
permissions:
|
||||
tbmc.admin:
|
||||
description: Gives access to /un- commands and /u admin commands
|
||||
tbmc.rainbow:
|
||||
description: Gives access to rainbow colors (/u c).
|
||||
name: TheButtonMCPlugin
|
||||
main: io.github.norbipeti.thebuttonmc.PluginMain
|
||||
version: 4.0
|
||||
commands:
|
||||
u:
|
||||
description: Auto-flair system. Accept or ignore flair.
|
||||
usage: "&cUsage: /u accept|ignore|opme&r"
|
||||
ooc:
|
||||
description: Send message in Out-of-Character.
|
||||
usage: "&cUsage: /ooc <message>&r"
|
||||
alias: nrp
|
||||
unlol:
|
||||
description: Unlaugh the last laugh.
|
||||
alias: unlaugh
|
||||
mwiki:
|
||||
description: Search the wiki.
|
||||
usage: "&cUsage: /mwiki [query]&r"
|
||||
dontrunthiscmd:
|
||||
tableflip:
|
||||
description: Flip a table.
|
||||
usage: "&cUsage: /tableflip [message]&r"
|
||||
unflip:
|
||||
description: Unflip a flip.
|
||||
usage: "&cUsage: /unflip [message]&r"
|
||||
chatonly:
|
||||
description: Enable chat only mode. Useful for chat clients.
|
||||
usage: "&cUsage: /chatonly&r"
|
||||
author: NorbiPeti
|
||||
depend: [Essentials, Towny, Minigames, Votifier, Factions, WorldGuard, WorldEdit, ProtocolLib, FastLogin, Vault, AuthMe]
|
||||
permissions:
|
||||
tbmc.admin:
|
||||
description: Gives access to /un- commands and /u admin commands
|
||||
tbmc.rainbow:
|
||||
description: Gives access to rainbow colors (/u c).
|
||||
|
|
File diff suppressed because it is too large
Load diff
File diff suppressed because it is too large
Load diff
|
@ -1,429 +1,408 @@
|
|||
package io.github.norbipeti.thebuttonmc;
|
||||
|
||||
import net.milkbowl.vault.chat.Chat;
|
||||
import net.milkbowl.vault.economy.Economy;
|
||||
import net.milkbowl.vault.permission.Permission;
|
||||
|
||||
import org.apache.commons.io.IOUtils;
|
||||
import org.bukkit.Bukkit;
|
||||
import org.bukkit.command.ConsoleCommandSender;
|
||||
import org.bukkit.configuration.InvalidConfigurationException;
|
||||
import org.bukkit.configuration.file.YamlConfiguration;
|
||||
import org.bukkit.entity.Player;
|
||||
import org.bukkit.plugin.RegisteredServiceProvider;
|
||||
import org.bukkit.plugin.java.JavaPlugin;
|
||||
import org.bukkit.scoreboard.Scoreboard;
|
||||
import org.htmlcleaner.HtmlCleaner;
|
||||
import org.htmlcleaner.TagNode;
|
||||
import org.json.JSONArray;
|
||||
import org.json.JSONObject;
|
||||
|
||||
import au.com.mineauz.minigames.mechanics.GameMechanics;
|
||||
|
||||
import com.palmergames.bukkit.towny.Towny;
|
||||
import com.palmergames.bukkit.towny.object.Nation;
|
||||
import com.palmergames.bukkit.towny.object.Town;
|
||||
import com.palmergames.bukkit.towny.object.TownyUniverse;
|
||||
|
||||
import java.io.*;
|
||||
import java.lang.String;
|
||||
import java.lang.reflect.Method;
|
||||
import java.net.MalformedURLException;
|
||||
import java.net.URL;
|
||||
import java.net.URLClassLoader;
|
||||
import java.net.URLConnection;
|
||||
import java.text.SimpleDateFormat;
|
||||
import java.util.ArrayList;
|
||||
import java.util.Calendar;
|
||||
import java.util.Collection;
|
||||
import java.util.Date;
|
||||
import java.util.TimeZone;
|
||||
|
||||
public class PluginMain extends JavaPlugin { // Translated to Java: 2015.07.15.
|
||||
// A user, which flair isn't obtainable:
|
||||
// https://www.reddit.com/r/thebutton/comments/31c32v/i_pressed_the_button_without_really_thinking/
|
||||
public static PluginMain Instance;
|
||||
public static ConsoleCommandSender Console; // 2015.08.12.
|
||||
public static Scoreboard SB;
|
||||
public final static String FlairThreadURL = "https://www.reddit.com/r/TheButtonMinecraft/comments/433ptk/autoflair_thread/";
|
||||
public TownyUniverse TU;
|
||||
public ArrayList<Town> Towns;
|
||||
public ArrayList<Nation> Nations;
|
||||
|
||||
// Fired when plugin is first enabled
|
||||
@Override
|
||||
public void onEnable() {
|
||||
try {
|
||||
System.out.println("Extracting necessary libraries...");
|
||||
final File[] libs = new File[] { new File(getDataFolder(),
|
||||
"htmlcleaner-2.16.jar") };
|
||||
for (final File lib : libs) {
|
||||
if (!lib.exists()) {
|
||||
JarUtils.extractFromJar(lib.getName(),
|
||||
lib.getAbsolutePath());
|
||||
}
|
||||
}
|
||||
for (final File lib : libs) {
|
||||
if (!lib.exists()) {
|
||||
getLogger().warning(
|
||||
"Failed to load plugin! Could not find lib: "
|
||||
+ lib.getName());
|
||||
Bukkit.getServer().getPluginManager().disablePlugin(this);
|
||||
return;
|
||||
}
|
||||
addClassPath(JarUtils.getJarUrl(lib));
|
||||
}
|
||||
} catch (final Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
|
||||
getServer().getPluginManager().registerEvents(new PlayerListener(),
|
||||
this);
|
||||
Commands comm = new Commands();
|
||||
this.getCommand("u").setExecutor(comm);
|
||||
this.getCommand("u").setUsage(
|
||||
this.getCommand("u").getUsage().replace('&', '§'));
|
||||
this.getCommand("nrp").setExecutor(comm);
|
||||
this.getCommand("nrp").setUsage(
|
||||
this.getCommand("nrp").getUsage().replace('&', '§'));
|
||||
this.getCommand("ooc").setExecutor(comm);
|
||||
this.getCommand("ooc").setUsage(
|
||||
this.getCommand("ooc").getUsage().replace('&', '§'));
|
||||
this.getCommand("unlol").setExecutor(comm);
|
||||
this.getCommand("unlaugh").setExecutor(comm);
|
||||
this.getCommand("mwiki").setExecutor(comm);
|
||||
this.getCommand("mwiki").setUsage(
|
||||
this.getCommand("mwiki").getUsage().replace('&', '§'));
|
||||
this.getCommand("tableflip").setExecutor(comm);
|
||||
this.getCommand("tableflip").setUsage(
|
||||
this.getCommand("tableflip").getUsage().replace('&', '§'));
|
||||
this.getCommand("unflip").setExecutor(comm);
|
||||
this.getCommand("unflip").setUsage(
|
||||
this.getCommand("unflip").getUsage().replace('&', '§'));
|
||||
this.getCommand("chatonly").setExecutor(comm);
|
||||
this.getCommand("chatonly").setUsage(
|
||||
this.getCommand("chatonly").getUsage().replace('&', '§'));
|
||||
Instance = this;
|
||||
Console = this.getServer().getConsoleSender();
|
||||
LoadFiles(false);
|
||||
|
||||
SB = PluginMain.Instance.getServer().getScoreboardManager()
|
||||
.getMainScoreboard(); // Main can be detected with @a[score_...]
|
||||
if (SB.getObjective("town") == null)
|
||||
SB.registerNewObjective("town", "dummy");
|
||||
if (SB.getObjective("nation") == null)
|
||||
SB.registerNewObjective("nation", "dummy");
|
||||
if (SB.getObjective("admin") == null)
|
||||
SB.registerNewObjective("admin", "dummy");
|
||||
if (SB.getObjective("mod") == null)
|
||||
SB.registerNewObjective("mod", "dummy");
|
||||
TU = ((Towny) Bukkit.getPluginManager().getPlugin("Towny"))
|
||||
.getTownyUniverse();
|
||||
Towns = new ArrayList<Town>(TU.getTownsMap().values());
|
||||
Nations = new ArrayList<Nation>(TU.getNationsMap().values());
|
||||
|
||||
setupChat();
|
||||
setupEconomy();
|
||||
setupPermissions();
|
||||
|
||||
GameMechanics.addGameMechanic(new CreativeGlobalMechanic());
|
||||
|
||||
Runnable r = new Runnable() {
|
||||
public void run() {
|
||||
ThreadMethod();
|
||||
}
|
||||
};
|
||||
Thread t = new Thread(r);
|
||||
t.start();
|
||||
r = new Runnable() {
|
||||
public void run() {
|
||||
AnnouncerThread.Run();
|
||||
}
|
||||
};
|
||||
t = new Thread(r);
|
||||
t.start();
|
||||
}
|
||||
|
||||
public Boolean stop = false;
|
||||
|
||||
// Fired when plugin is disabled
|
||||
@Override
|
||||
public void onDisable() {
|
||||
SaveFiles(); // 2015.08.09.
|
||||
stop = true;
|
||||
}
|
||||
|
||||
private void ThreadMethod() {
|
||||
while (!stop) {
|
||||
try {
|
||||
String body = DownloadString(FlairThreadURL
|
||||
+ ".json?limit=1000");
|
||||
JSONArray json = new JSONArray(body).getJSONObject(1)
|
||||
.getJSONObject("data").getJSONArray("children");
|
||||
for (Object obj : json) {
|
||||
JSONObject item = (JSONObject) obj;
|
||||
String author = item.getJSONObject("data").getString(
|
||||
"author");
|
||||
String ign = item.getJSONObject("data").getString("body");
|
||||
int start = ign.indexOf("IGN:") + "IGN:".length();
|
||||
if (start == -1 + "IGN:".length()) // +length: 2015.08.10.
|
||||
continue; // 2015.08.09.
|
||||
int end = ign.indexOf(' ', start);
|
||||
if (end == -1 || end == start)
|
||||
end = ign.indexOf('\n', start); // 2015.07.15.
|
||||
if (end == -1 || end == start)
|
||||
ign = ign.substring(start);
|
||||
else
|
||||
ign = ign.substring(start, end);
|
||||
ign = ign.trim();
|
||||
MaybeOfflinePlayer mp = MaybeOfflinePlayer.GetFromName(ign);
|
||||
if (mp == null)
|
||||
continue;
|
||||
if (!JoinedBefore(mp, 2015, 6, 5))
|
||||
continue;
|
||||
if (!mp.UserNames.contains(author))
|
||||
mp.UserNames.add(author);
|
||||
if (mp.FlairState.equals(FlairStates.NoComment)) {
|
||||
mp.FlairState = FlairStates.Commented;
|
||||
ConfirmUserMessage(mp);
|
||||
}
|
||||
try {
|
||||
Thread.sleep(10);
|
||||
} catch (InterruptedException ex) {
|
||||
Thread.currentThread().interrupt();
|
||||
}
|
||||
}
|
||||
try {
|
||||
Thread.sleep(10000);
|
||||
} catch (InterruptedException ex) {
|
||||
Thread.currentThread().interrupt();
|
||||
}
|
||||
} catch (Exception e) {
|
||||
// System.out.println("Error!\n" + e);
|
||||
LastException = e; // 2015.08.09.
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public void DownloadFlair(MaybeOfflinePlayer mp)
|
||||
throws MalformedURLException, IOException {
|
||||
String[] flairdata = DownloadString(
|
||||
"http://karmadecay.com/thebutton-data.php?users=" + mp.UserName)
|
||||
.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);
|
||||
}
|
||||
|
||||
public static Exception LastException; // 2015.08.09.
|
||||
|
||||
public String DownloadString(String urlstr) throws MalformedURLException,
|
||||
IOException {
|
||||
URL url = new URL(urlstr);
|
||||
URLConnection con = url.openConnection();
|
||||
con.setRequestProperty("User-Agent", "TheButtonAutoFlair");
|
||||
InputStream in = con.getInputStream();
|
||||
String encoding = con.getContentEncoding();
|
||||
encoding = encoding == null ? "UTF-8" : encoding;
|
||||
String body = IOUtils.toString(in, encoding);
|
||||
in.close();
|
||||
return body;
|
||||
}
|
||||
|
||||
private void SetFlair(MaybeOfflinePlayer p, String text, String flairclass,
|
||||
String username) {
|
||||
p.UserName = username;
|
||||
p.FlairState = FlairStates.Recognised;
|
||||
switch (flairclass) {
|
||||
case "cheater":
|
||||
p.SetFlair((short) 0x5, true);
|
||||
return;
|
||||
case "unknown":
|
||||
if (text.equals("-1")) // If true, only non-presser/can't press; if
|
||||
// false, any flair (but we can still detect
|
||||
// can't press)
|
||||
{
|
||||
try {
|
||||
if (CheckForJoinDate(p)) {
|
||||
p.SetFlair(MaybeOfflinePlayer.FlairTimeNonPresser);
|
||||
} else {
|
||||
p.SetFlair(MaybeOfflinePlayer.FlairTimeCantPress);
|
||||
}
|
||||
} catch (Exception e) {
|
||||
p.FlairState = FlairStates.Commented; // Flair unknown
|
||||
p.SetFlair(MaybeOfflinePlayer.FlairTimeNone);
|
||||
e.printStackTrace();
|
||||
}
|
||||
} else {
|
||||
try {
|
||||
if (CheckForJoinDate(p)) {
|
||||
p.FlairState = FlairStates.Commented; // Flair unknown
|
||||
p.SetFlair(MaybeOfflinePlayer.FlairTimeNone);
|
||||
} else {
|
||||
p.SetFlair(MaybeOfflinePlayer.FlairTimeCantPress);
|
||||
}
|
||||
} catch (Exception e) {
|
||||
p.FlairState = FlairStates.Commented; // Flair unknown
|
||||
p.SetFlair(MaybeOfflinePlayer.FlairTimeNone);
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
return;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
p.SetFlair(Short.parseShort(text));
|
||||
}
|
||||
|
||||
public static boolean CheckForJoinDate(MaybeOfflinePlayer mp)
|
||||
throws Exception {
|
||||
return JoinedBefore(mp, 2015, 4, 1);
|
||||
}
|
||||
|
||||
public static boolean JoinedBefore(MaybeOfflinePlayer 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(MaybeOfflinePlayer mp) {
|
||||
Player p = Bukkit.getPlayer(mp.UUID);
|
||||
if (mp.FlairState.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 Collection<? extends Player> GetPlayers() {
|
||||
return Instance.getServer().getOnlinePlayers();
|
||||
}
|
||||
|
||||
public static ArrayList<String> AnnounceMessages = new ArrayList<>();
|
||||
public static int AnnounceTime = 15 * 60 * 1000;
|
||||
|
||||
public static void LoadFiles(boolean reload) {
|
||||
if (reload) {
|
||||
System.out
|
||||
.println("The Button Minecraft plugin cleanup for reloading...");
|
||||
MaybeOfflinePlayer.AllPlayers.clear();
|
||||
AnnounceMessages.clear();
|
||||
}
|
||||
System.out.println("Loading files for The Button Minecraft plugin...");
|
||||
try {
|
||||
File file = new File("thebuttonmc.yml");
|
||||
if (file.exists()) {
|
||||
YamlConfiguration yc = new YamlConfiguration();
|
||||
yc.load(file);
|
||||
MaybeOfflinePlayer.Load(yc);
|
||||
PlayerListener.NotificationSound = yc
|
||||
.getString("notificationsound");
|
||||
PlayerListener.NotificationPitch = yc
|
||||
.getDouble("notificationpitch");
|
||||
AnnounceTime = yc.getInt("announcetime");
|
||||
AnnounceMessages.addAll(yc.getStringList("announcements"));
|
||||
PlayerListener.AlphaDeaths = yc.getInt("alphadeaths");
|
||||
}
|
||||
System.out.println("The Button Minecraft plugin loaded files!");
|
||||
} catch (IOException e) {
|
||||
System.out.println("Error!\n" + e);
|
||||
LastException = e;
|
||||
} catch (InvalidConfigurationException e) {
|
||||
System.out.println("Error!\n" + e);
|
||||
LastException = e;
|
||||
}
|
||||
}
|
||||
|
||||
public static void SaveFiles() {
|
||||
System.out.println("Saving files for The Button Minecraft plugin...");
|
||||
try {
|
||||
File file = new File("thebuttonmc.yml");
|
||||
YamlConfiguration yc = new YamlConfiguration();
|
||||
MaybeOfflinePlayer.Save(yc);
|
||||
yc.set("notificationsound", PlayerListener.NotificationSound);
|
||||
yc.set("notificationpitch", PlayerListener.NotificationPitch);
|
||||
yc.set("announcetime", AnnounceTime);
|
||||
yc.set("announcements", AnnounceMessages);
|
||||
yc.set("alphadeaths", PlayerListener.AlphaDeaths);
|
||||
yc.save(file);
|
||||
System.out.println("The Button Minecraft plugin saved files!");
|
||||
} catch (IOException e) {
|
||||
System.out.println("Error!\n" + e);
|
||||
LastException = e;
|
||||
}
|
||||
}
|
||||
|
||||
private void addClassPath(final URL url) throws IOException {
|
||||
final URLClassLoader sysloader = (URLClassLoader) ClassLoader
|
||||
.getSystemClassLoader();
|
||||
final Class<URLClassLoader> sysclass = URLClassLoader.class;
|
||||
try {
|
||||
final Method method = sysclass.getDeclaredMethod("addURL",
|
||||
new Class[] { URL.class });
|
||||
method.setAccessible(true);
|
||||
method.invoke(sysloader, new Object[] { url });
|
||||
} catch (final Throwable t) {
|
||||
t.printStackTrace();
|
||||
throw new IOException("Error adding " + url
|
||||
+ " to system classloader");
|
||||
}
|
||||
}
|
||||
|
||||
public static Permission permission = null;
|
||||
public static Economy economy = null;
|
||||
public static Chat chat = null;
|
||||
|
||||
private boolean setupPermissions() {
|
||||
RegisteredServiceProvider<Permission> permissionProvider = getServer()
|
||||
.getServicesManager().getRegistration(
|
||||
net.milkbowl.vault.permission.Permission.class);
|
||||
if (permissionProvider != null) {
|
||||
permission = permissionProvider.getProvider();
|
||||
}
|
||||
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() {
|
||||
RegisteredServiceProvider<Economy> economyProvider = getServer()
|
||||
.getServicesManager().getRegistration(
|
||||
net.milkbowl.vault.economy.Economy.class);
|
||||
if (economyProvider != null) {
|
||||
economy = economyProvider.getProvider();
|
||||
}
|
||||
|
||||
return (economy != null);
|
||||
}
|
||||
}
|
||||
package io.github.norbipeti.thebuttonmc;
|
||||
|
||||
import net.milkbowl.vault.chat.Chat;
|
||||
import net.milkbowl.vault.economy.Economy;
|
||||
import net.milkbowl.vault.permission.Permission;
|
||||
|
||||
import org.apache.commons.io.IOUtils;
|
||||
import org.bukkit.Bukkit;
|
||||
import org.bukkit.command.ConsoleCommandSender;
|
||||
import org.bukkit.configuration.InvalidConfigurationException;
|
||||
import org.bukkit.configuration.file.YamlConfiguration;
|
||||
import org.bukkit.entity.Player;
|
||||
import org.bukkit.plugin.RegisteredServiceProvider;
|
||||
import org.bukkit.plugin.java.JavaPlugin;
|
||||
import org.bukkit.scoreboard.Scoreboard;
|
||||
import org.htmlcleaner.HtmlCleaner;
|
||||
import org.htmlcleaner.TagNode;
|
||||
import org.json.JSONArray;
|
||||
import org.json.JSONObject;
|
||||
|
||||
import au.com.mineauz.minigames.mechanics.GameMechanics;
|
||||
|
||||
import com.palmergames.bukkit.towny.Towny;
|
||||
import com.palmergames.bukkit.towny.object.Nation;
|
||||
import com.palmergames.bukkit.towny.object.Town;
|
||||
import com.palmergames.bukkit.towny.object.TownyUniverse;
|
||||
|
||||
import io.github.norbipeti.thebuttonmc.commands.TBMCCommandBase;
|
||||
|
||||
import java.io.*;
|
||||
import java.lang.String;
|
||||
import java.lang.reflect.Method;
|
||||
import java.net.MalformedURLException;
|
||||
import java.net.URL;
|
||||
import java.net.URLClassLoader;
|
||||
import java.net.URLConnection;
|
||||
import java.text.SimpleDateFormat;
|
||||
import java.util.ArrayList;
|
||||
import java.util.Calendar;
|
||||
import java.util.Collection;
|
||||
import java.util.Date;
|
||||
import java.util.TimeZone;
|
||||
|
||||
public class PluginMain extends JavaPlugin { // Translated to Java: 2015.07.15.
|
||||
// A user, which flair isn't obtainable:
|
||||
// https://www.reddit.com/r/thebutton/comments/31c32v/i_pressed_the_button_without_really_thinking/
|
||||
public static PluginMain Instance;
|
||||
public static ConsoleCommandSender Console; // 2015.08.12.
|
||||
public static Scoreboard SB;
|
||||
public final static String FlairThreadURL = "https://www.reddit.com/r/TheButtonMinecraft/comments/433ptk/autoflair_thread/";
|
||||
public TownyUniverse TU;
|
||||
public ArrayList<Town> Towns;
|
||||
public ArrayList<Nation> Nations;
|
||||
|
||||
// Fired when plugin is first enabled
|
||||
@Override
|
||||
public void onEnable() {
|
||||
try {
|
||||
System.out.println("Extracting necessary libraries...");
|
||||
final File[] libs = new File[] { new File(getDataFolder(),
|
||||
"htmlcleaner-2.16.jar") };
|
||||
for (final File lib : libs) {
|
||||
if (!lib.exists()) {
|
||||
JarUtils.extractFromJar(lib.getName(),
|
||||
lib.getAbsolutePath());
|
||||
}
|
||||
}
|
||||
for (final File lib : libs) {
|
||||
if (!lib.exists()) {
|
||||
getLogger().warning(
|
||||
"Failed to load plugin! Could not find lib: "
|
||||
+ lib.getName());
|
||||
Bukkit.getServer().getPluginManager().disablePlugin(this);
|
||||
return;
|
||||
}
|
||||
addClassPath(JarUtils.getJarUrl(lib));
|
||||
}
|
||||
} catch (final Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
|
||||
getServer().getPluginManager().registerEvents(new PlayerListener(),
|
||||
this);
|
||||
TBMCCommandBase.RegisterCommands(this);
|
||||
Instance = this;
|
||||
Console = this.getServer().getConsoleSender();
|
||||
LoadFiles(false);
|
||||
|
||||
SB = PluginMain.Instance.getServer().getScoreboardManager()
|
||||
.getMainScoreboard(); // Main can be detected with @a[score_...]
|
||||
if (SB.getObjective("town") == null)
|
||||
SB.registerNewObjective("town", "dummy");
|
||||
if (SB.getObjective("nation") == null)
|
||||
SB.registerNewObjective("nation", "dummy");
|
||||
if (SB.getObjective("admin") == null)
|
||||
SB.registerNewObjective("admin", "dummy");
|
||||
if (SB.getObjective("mod") == null)
|
||||
SB.registerNewObjective("mod", "dummy");
|
||||
TU = ((Towny) Bukkit.getPluginManager().getPlugin("Towny"))
|
||||
.getTownyUniverse();
|
||||
Towns = new ArrayList<Town>(TU.getTownsMap().values());
|
||||
Nations = new ArrayList<Nation>(TU.getNationsMap().values());
|
||||
|
||||
setupChat();
|
||||
setupEconomy();
|
||||
setupPermissions();
|
||||
|
||||
GameMechanics.addGameMechanic(new CreativeGlobalMechanic());
|
||||
|
||||
Runnable r = new Runnable() {
|
||||
public void run() {
|
||||
ThreadMethod();
|
||||
}
|
||||
};
|
||||
Thread t = new Thread(r);
|
||||
t.start();
|
||||
r = new Runnable() {
|
||||
public void run() {
|
||||
AnnouncerThread.Run();
|
||||
}
|
||||
};
|
||||
t = new Thread(r);
|
||||
t.start();
|
||||
}
|
||||
|
||||
public Boolean stop = false;
|
||||
|
||||
// Fired when plugin is disabled
|
||||
@Override
|
||||
public void onDisable() {
|
||||
SaveFiles(); // 2015.08.09.
|
||||
stop = true;
|
||||
}
|
||||
|
||||
private void ThreadMethod() {
|
||||
while (!stop) {
|
||||
try {
|
||||
String body = DownloadString(FlairThreadURL
|
||||
+ ".json?limit=1000");
|
||||
JSONArray json = new JSONArray(body).getJSONObject(1)
|
||||
.getJSONObject("data").getJSONArray("children");
|
||||
for (Object obj : json) {
|
||||
JSONObject item = (JSONObject) obj;
|
||||
String author = item.getJSONObject("data").getString(
|
||||
"author");
|
||||
String ign = item.getJSONObject("data").getString("body");
|
||||
int start = ign.indexOf("IGN:") + "IGN:".length();
|
||||
if (start == -1 + "IGN:".length()) // +length: 2015.08.10.
|
||||
continue; // 2015.08.09.
|
||||
int end = ign.indexOf(' ', start);
|
||||
if (end == -1 || end == start)
|
||||
end = ign.indexOf('\n', start); // 2015.07.15.
|
||||
if (end == -1 || end == start)
|
||||
ign = ign.substring(start);
|
||||
else
|
||||
ign = ign.substring(start, end);
|
||||
ign = ign.trim();
|
||||
MaybeOfflinePlayer mp = MaybeOfflinePlayer.GetFromName(ign);
|
||||
if (mp == null)
|
||||
continue;
|
||||
if (!JoinedBefore(mp, 2015, 6, 5))
|
||||
continue;
|
||||
if (!mp.UserNames.contains(author))
|
||||
mp.UserNames.add(author);
|
||||
if (mp.FlairState.equals(FlairStates.NoComment)) {
|
||||
mp.FlairState = FlairStates.Commented;
|
||||
ConfirmUserMessage(mp);
|
||||
}
|
||||
try {
|
||||
Thread.sleep(10);
|
||||
} catch (InterruptedException ex) {
|
||||
Thread.currentThread().interrupt();
|
||||
}
|
||||
}
|
||||
try {
|
||||
Thread.sleep(10000);
|
||||
} catch (InterruptedException ex) {
|
||||
Thread.currentThread().interrupt();
|
||||
}
|
||||
} catch (Exception e) {
|
||||
// System.out.println("Error!\n" + e);
|
||||
LastException = e; // 2015.08.09.
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public void DownloadFlair(MaybeOfflinePlayer mp)
|
||||
throws MalformedURLException, IOException {
|
||||
String[] flairdata = DownloadString(
|
||||
"http://karmadecay.com/thebutton-data.php?users=" + mp.UserName)
|
||||
.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);
|
||||
}
|
||||
|
||||
public static Exception LastException; // 2015.08.09.
|
||||
|
||||
public String DownloadString(String urlstr) throws MalformedURLException,
|
||||
IOException {
|
||||
URL url = new URL(urlstr);
|
||||
URLConnection con = url.openConnection();
|
||||
con.setRequestProperty("User-Agent", "TheButtonAutoFlair");
|
||||
InputStream in = con.getInputStream();
|
||||
String encoding = con.getContentEncoding();
|
||||
encoding = encoding == null ? "UTF-8" : encoding;
|
||||
String body = IOUtils.toString(in, encoding);
|
||||
in.close();
|
||||
return body;
|
||||
}
|
||||
|
||||
private void SetFlair(MaybeOfflinePlayer p, String text, String flairclass,
|
||||
String username) {
|
||||
p.UserName = username;
|
||||
p.FlairState = FlairStates.Recognised;
|
||||
switch (flairclass) {
|
||||
case "cheater":
|
||||
p.SetFlair((short) 0x5, true);
|
||||
return;
|
||||
case "unknown":
|
||||
if (text.equals("-1")) // If true, only non-presser/can't press; if
|
||||
// false, any flair (but we can still detect
|
||||
// can't press)
|
||||
{
|
||||
try {
|
||||
if (CheckForJoinDate(p)) {
|
||||
p.SetFlair(MaybeOfflinePlayer.FlairTimeNonPresser);
|
||||
} else {
|
||||
p.SetFlair(MaybeOfflinePlayer.FlairTimeCantPress);
|
||||
}
|
||||
} catch (Exception e) {
|
||||
p.FlairState = FlairStates.Commented; // Flair unknown
|
||||
p.SetFlair(MaybeOfflinePlayer.FlairTimeNone);
|
||||
e.printStackTrace();
|
||||
}
|
||||
} else {
|
||||
try {
|
||||
if (CheckForJoinDate(p)) {
|
||||
p.FlairState = FlairStates.Commented; // Flair unknown
|
||||
p.SetFlair(MaybeOfflinePlayer.FlairTimeNone);
|
||||
} else {
|
||||
p.SetFlair(MaybeOfflinePlayer.FlairTimeCantPress);
|
||||
}
|
||||
} catch (Exception e) {
|
||||
p.FlairState = FlairStates.Commented; // Flair unknown
|
||||
p.SetFlair(MaybeOfflinePlayer.FlairTimeNone);
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
return;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
p.SetFlair(Short.parseShort(text));
|
||||
}
|
||||
|
||||
public static boolean CheckForJoinDate(MaybeOfflinePlayer mp)
|
||||
throws Exception {
|
||||
return JoinedBefore(mp, 2015, 4, 1);
|
||||
}
|
||||
|
||||
public static boolean JoinedBefore(MaybeOfflinePlayer 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(MaybeOfflinePlayer mp) {
|
||||
Player p = Bukkit.getPlayer(mp.UUID);
|
||||
if (mp.FlairState.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 Collection<? extends Player> GetPlayers() {
|
||||
return Instance.getServer().getOnlinePlayers();
|
||||
}
|
||||
|
||||
public static ArrayList<String> AnnounceMessages = new ArrayList<>();
|
||||
public static int AnnounceTime = 15 * 60 * 1000;
|
||||
|
||||
public static void LoadFiles(boolean reload) {
|
||||
if (reload) {
|
||||
System.out
|
||||
.println("The Button Minecraft plugin cleanup for reloading...");
|
||||
MaybeOfflinePlayer.AllPlayers.clear();
|
||||
AnnounceMessages.clear();
|
||||
}
|
||||
System.out.println("Loading files for The Button Minecraft plugin...");
|
||||
try {
|
||||
File file = new File("thebuttonmc.yml");
|
||||
if (file.exists()) {
|
||||
YamlConfiguration yc = new YamlConfiguration();
|
||||
yc.load(file);
|
||||
MaybeOfflinePlayer.Load(yc);
|
||||
PlayerListener.NotificationSound = yc
|
||||
.getString("notificationsound");
|
||||
PlayerListener.NotificationPitch = yc
|
||||
.getDouble("notificationpitch");
|
||||
AnnounceTime = yc.getInt("announcetime");
|
||||
AnnounceMessages.addAll(yc.getStringList("announcements"));
|
||||
PlayerListener.AlphaDeaths = yc.getInt("alphadeaths");
|
||||
}
|
||||
System.out.println("The Button Minecraft plugin loaded files!");
|
||||
} catch (IOException e) {
|
||||
System.out.println("Error!\n" + e);
|
||||
LastException = e;
|
||||
} catch (InvalidConfigurationException e) {
|
||||
System.out.println("Error!\n" + e);
|
||||
LastException = e;
|
||||
}
|
||||
}
|
||||
|
||||
public static void SaveFiles() {
|
||||
System.out.println("Saving files for The Button Minecraft plugin...");
|
||||
try {
|
||||
File file = new File("thebuttonmc.yml");
|
||||
YamlConfiguration yc = new YamlConfiguration();
|
||||
MaybeOfflinePlayer.Save(yc);
|
||||
yc.set("notificationsound", PlayerListener.NotificationSound);
|
||||
yc.set("notificationpitch", PlayerListener.NotificationPitch);
|
||||
yc.set("announcetime", AnnounceTime);
|
||||
yc.set("announcements", AnnounceMessages);
|
||||
yc.set("alphadeaths", PlayerListener.AlphaDeaths);
|
||||
yc.save(file);
|
||||
System.out.println("The Button Minecraft plugin saved files!");
|
||||
} catch (IOException e) {
|
||||
System.out.println("Error!\n" + e);
|
||||
LastException = e;
|
||||
}
|
||||
}
|
||||
|
||||
private void addClassPath(final URL url) throws IOException {
|
||||
final URLClassLoader sysloader = (URLClassLoader) ClassLoader
|
||||
.getSystemClassLoader();
|
||||
final Class<URLClassLoader> sysclass = URLClassLoader.class;
|
||||
try {
|
||||
final Method method = sysclass.getDeclaredMethod("addURL",
|
||||
new Class[] { URL.class });
|
||||
method.setAccessible(true);
|
||||
method.invoke(sysloader, new Object[] { url });
|
||||
} catch (final Throwable t) {
|
||||
t.printStackTrace();
|
||||
throw new IOException("Error adding " + url
|
||||
+ " to system classloader");
|
||||
}
|
||||
}
|
||||
|
||||
public static Permission permission = null;
|
||||
public static Economy economy = null;
|
||||
public static Chat chat = null;
|
||||
|
||||
private boolean setupPermissions() {
|
||||
RegisteredServiceProvider<Permission> permissionProvider = getServer()
|
||||
.getServicesManager().getRegistration(
|
||||
net.milkbowl.vault.permission.Permission.class);
|
||||
if (permissionProvider != null) {
|
||||
permission = permissionProvider.getProvider();
|
||||
}
|
||||
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() {
|
||||
RegisteredServiceProvider<Economy> economyProvider = getServer()
|
||||
.getServicesManager().getRegistration(
|
||||
net.milkbowl.vault.economy.Economy.class);
|
||||
if (economyProvider != null) {
|
||||
economy = economyProvider.getProvider();
|
||||
}
|
||||
|
||||
return (economy != null);
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue