2011-10-08 22:03:44 +02:00
|
|
|
package com.massivecraft.factions;
|
|
|
|
|
2018-12-27 02:22:06 +01:00
|
|
|
import ch.njol.skript.Skript;
|
|
|
|
import ch.njol.skript.SkriptAddon;
|
2014-10-25 20:22:41 +02:00
|
|
|
import com.google.gson.GsonBuilder;
|
|
|
|
import com.google.gson.reflect.TypeToken;
|
2012-01-08 00:27:03 +01:00
|
|
|
import com.massivecraft.factions.cmd.CmdAutoHelp;
|
|
|
|
import com.massivecraft.factions.cmd.FCmdRoot;
|
2014-04-05 22:42:01 +02:00
|
|
|
import com.massivecraft.factions.integration.Econ;
|
|
|
|
import com.massivecraft.factions.integration.Worldguard;
|
2014-12-05 06:04:00 +01:00
|
|
|
import com.massivecraft.factions.integration.dynmap.EngineDynmap;
|
2014-04-04 20:55:21 +02:00
|
|
|
import com.massivecraft.factions.listeners.*;
|
2019-08-04 22:23:40 +02:00
|
|
|
import com.massivecraft.factions.shop.ShopClickPersistence;
|
|
|
|
import com.massivecraft.factions.shop.ShopConfig;
|
2011-10-08 22:03:44 +02:00
|
|
|
import com.massivecraft.factions.struct.ChatMode;
|
2019-08-05 14:05:22 +02:00
|
|
|
import com.massivecraft.factions.util.Placeholder;
|
2014-10-25 20:22:41 +02:00
|
|
|
import com.massivecraft.factions.util.*;
|
2018-04-03 16:38:11 +02:00
|
|
|
import com.massivecraft.factions.util.Particles.ReflectionUtils;
|
2018-12-09 03:13:18 +01:00
|
|
|
import com.massivecraft.factions.zcore.CommandVisibility;
|
|
|
|
import com.massivecraft.factions.zcore.MCommand;
|
2011-10-08 22:03:44 +02:00
|
|
|
import com.massivecraft.factions.zcore.MPlugin;
|
2018-01-05 00:44:39 +01:00
|
|
|
import com.massivecraft.factions.zcore.fperms.Access;
|
2018-02-06 04:46:43 +01:00
|
|
|
import com.massivecraft.factions.zcore.fperms.Permissable;
|
2018-02-03 21:49:04 +01:00
|
|
|
import com.massivecraft.factions.zcore.fperms.PermissableAction;
|
2019-05-14 15:27:57 +02:00
|
|
|
import com.massivecraft.factions.zcore.fupgrades.*;
|
2012-03-11 17:41:56 +01:00
|
|
|
import com.massivecraft.factions.zcore.util.TextUtil;
|
2018-03-26 23:43:15 +02:00
|
|
|
import net.milkbowl.vault.economy.Economy;
|
2014-11-13 20:49:13 +01:00
|
|
|
import net.milkbowl.vault.permission.Permission;
|
2018-03-26 23:43:15 +02:00
|
|
|
import org.bukkit.*;
|
2014-04-04 20:55:21 +02:00
|
|
|
import org.bukkit.command.Command;
|
|
|
|
import org.bukkit.command.CommandSender;
|
2018-03-26 23:43:15 +02:00
|
|
|
import org.bukkit.entity.ArmorStand;
|
|
|
|
import org.bukkit.entity.EntityType;
|
2014-04-04 20:55:21 +02:00
|
|
|
import org.bukkit.entity.Player;
|
2018-12-09 03:13:18 +01:00
|
|
|
import org.bukkit.event.Listener;
|
2014-04-04 20:55:21 +02:00
|
|
|
import org.bukkit.event.player.AsyncPlayerChatEvent;
|
2018-09-09 19:40:45 +02:00
|
|
|
import org.bukkit.inventory.Inventory;
|
2018-03-26 23:43:15 +02:00
|
|
|
import org.bukkit.inventory.ItemStack;
|
|
|
|
import org.bukkit.inventory.meta.ItemMeta;
|
2016-05-31 01:22:45 +02:00
|
|
|
import org.bukkit.plugin.Plugin;
|
2014-11-13 20:49:13 +01:00
|
|
|
import org.bukkit.plugin.RegisteredServiceProvider;
|
2014-04-04 20:55:21 +02:00
|
|
|
|
2018-11-12 03:05:57 +01:00
|
|
|
import java.io.*;
|
2014-04-04 20:55:21 +02:00
|
|
|
import java.lang.reflect.Modifier;
|
|
|
|
import java.lang.reflect.Type;
|
2018-03-26 23:43:15 +02:00
|
|
|
import java.util.*;
|
2019-05-15 09:10:10 +02:00
|
|
|
import java.util.concurrent.atomic.AtomicReference;
|
2014-04-04 20:55:21 +02:00
|
|
|
import java.util.logging.Level;
|
2018-12-09 03:13:18 +01:00
|
|
|
import java.util.stream.Collectors;
|
2011-10-08 22:03:44 +02:00
|
|
|
|
2018-03-26 23:43:15 +02:00
|
|
|
|
2019-08-02 23:01:33 +02:00
|
|
|
public class P extends MPlugin {
|
2014-08-05 17:17:27 +02:00
|
|
|
|
2019-03-03 04:51:21 +01:00
|
|
|
// Our single plugin instance.
|
|
|
|
// Single 4 life.
|
2019-08-02 23:01:33 +02:00
|
|
|
public static P p;
|
2019-03-03 04:51:21 +01:00
|
|
|
public static Permission perms = null;
|
2019-03-04 22:46:39 +01:00
|
|
|
// This plugin sets the boolean true when fully enabled.
|
|
|
|
// Plugins can check this boolean while hooking in have
|
|
|
|
// a green light to use the api.
|
2019-03-04 22:39:29 +01:00
|
|
|
public static boolean startupFinished = false;
|
2019-06-04 00:34:10 +02:00
|
|
|
private FactionsPlayerListener factionsPlayerListener;
|
2019-03-04 22:39:29 +01:00
|
|
|
|
2019-04-07 17:01:12 +02:00
|
|
|
|
2019-03-03 04:51:21 +01:00
|
|
|
public boolean PlaceholderApi;
|
|
|
|
// Commands
|
|
|
|
public FCmdRoot cmdBase;
|
|
|
|
public CmdAutoHelp cmdAutoHelp;
|
|
|
|
public boolean mc17 = false;
|
|
|
|
public boolean mc18 = false;
|
2019-08-05 14:05:22 +02:00
|
|
|
public boolean mc112 = false;
|
2019-03-03 04:51:21 +01:00
|
|
|
public boolean mc113 = false;
|
2019-05-15 09:10:10 +02:00
|
|
|
public boolean mc114 = false;
|
2019-03-03 04:51:21 +01:00
|
|
|
public boolean useNonPacketParticles = false;
|
|
|
|
public boolean factionsFlight = false;
|
|
|
|
SkriptAddon skriptAddon;
|
|
|
|
private boolean locked = false;
|
2019-06-02 20:28:42 +02:00
|
|
|
private boolean spam = false;
|
2019-03-03 04:51:21 +01:00
|
|
|
private Integer AutoLeaveTask = null;
|
|
|
|
private boolean hookedPlayervaults;
|
|
|
|
private ClipPlaceholderAPIManager clipPlaceholderAPIManager;
|
|
|
|
private boolean mvdwPlaceholderAPIManager = false;
|
|
|
|
private Listener[] eventsListener;
|
2019-06-02 07:52:34 +02:00
|
|
|
public static Economy econ = null;
|
2019-03-03 04:51:21 +01:00
|
|
|
|
2019-08-02 23:01:33 +02:00
|
|
|
public P() {
|
|
|
|
p = this;
|
2019-03-03 04:51:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public boolean getLocked() {
|
|
|
|
return this.locked;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setLocked(boolean val) {
|
|
|
|
this.locked = val;
|
|
|
|
this.setAutoSave(val);
|
|
|
|
}
|
|
|
|
|
2019-06-02 20:28:42 +02:00
|
|
|
public boolean getSpam() {
|
|
|
|
return this.spam;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setSpam(boolean val) {
|
|
|
|
this.spam = val;
|
|
|
|
this.setAutoSave(val);
|
|
|
|
}
|
|
|
|
|
2019-03-03 04:51:21 +01:00
|
|
|
public void playSoundForAll(String sound) {
|
|
|
|
for (Player pl : Bukkit.getOnlinePlayers()) {
|
|
|
|
playSound(pl, sound);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void playSoundForAll(List<String> sounds) {
|
|
|
|
for (Player pl : Bukkit.getOnlinePlayers()) {
|
|
|
|
playSound(pl, sounds);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void playSound(Player p, List<String> sounds) {
|
|
|
|
for (String sound : sounds) {
|
|
|
|
playSound(p, sound);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void playSound(Player p, String sound) {
|
|
|
|
float pitch = Float.valueOf(sound.split(":")[1]);
|
|
|
|
sound = sound.split(":")[0];
|
|
|
|
p.playSound(p.getLocation(), Sound.valueOf(sound), pitch, 5.0F);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onEnable() {
|
|
|
|
log("==== Setup ====");
|
|
|
|
|
|
|
|
|
|
|
|
// Vault dependency check.
|
|
|
|
if (getServer().getPluginManager().getPlugin("Vault") == null) {
|
|
|
|
log("Vault is not present, the plugin will not run properly.");
|
2019-08-02 23:01:33 +02:00
|
|
|
getServer().getPluginManager().disablePlugin(p);
|
2019-03-03 04:51:21 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int version = Integer.parseInt(ReflectionUtils.PackageType.getServerVersion().split("_")[1]);
|
2019-05-15 09:10:10 +02:00
|
|
|
switch (version) {
|
|
|
|
case 7:
|
2019-08-02 23:01:33 +02:00
|
|
|
P.p.log("Minecraft Version 1.7 found, disabling banners, itemflags inside GUIs, and Titles.");
|
2019-05-15 09:10:10 +02:00
|
|
|
mc17 = true;
|
|
|
|
break;
|
|
|
|
case 8:
|
2019-08-02 23:01:33 +02:00
|
|
|
P.p.log("Minecraft Version 1.8 found, Title Fadeouttime etc will not be configurable.");
|
2019-05-15 09:10:10 +02:00
|
|
|
mc18 = true;
|
|
|
|
break;
|
2019-08-05 14:05:22 +02:00
|
|
|
case 12:
|
|
|
|
mc112 = true;
|
|
|
|
break;
|
2019-05-15 09:10:10 +02:00
|
|
|
case 13:
|
2019-08-02 23:01:33 +02:00
|
|
|
P.p.log("Minecraft Version 1.13 found, New Items will be used.");
|
2019-05-15 09:10:10 +02:00
|
|
|
mc113 = true;
|
|
|
|
break;
|
|
|
|
case 14:
|
2019-08-02 23:01:33 +02:00
|
|
|
P.p.log("Minecraft Version 1.14 found.");
|
2019-05-15 09:10:10 +02:00
|
|
|
mc114 = true;
|
|
|
|
break;
|
2019-03-03 04:51:21 +01:00
|
|
|
}
|
|
|
|
migrateFPlayerLeaders();
|
|
|
|
log("==== End Setup ====");
|
|
|
|
|
|
|
|
if (!preEnable()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this.loadSuccessful = false;
|
|
|
|
|
|
|
|
saveDefaultConfig();
|
|
|
|
|
|
|
|
// Load Conf from disk
|
|
|
|
Conf.load();
|
|
|
|
com.massivecraft.factions.integration.Essentials.setup();
|
|
|
|
hookedPlayervaults = setupPlayervaults();
|
|
|
|
FPlayers.getInstance().load();
|
|
|
|
Factions.getInstance().load();
|
|
|
|
|
|
|
|
for (FPlayer fPlayer : FPlayers.getInstance().getAllFPlayers()) {
|
|
|
|
Faction faction = Factions.getInstance().getFactionById(fPlayer.getFactionId());
|
|
|
|
if (faction == null) {
|
|
|
|
log("Invalid faction id on " + fPlayer.getName() + ":" + fPlayer.getFactionId());
|
|
|
|
fPlayer.resetFactionData(false);
|
|
|
|
continue;
|
|
|
|
}
|
2019-05-19 22:09:00 +02:00
|
|
|
if (fPlayer.isAlt()) {
|
|
|
|
faction.addAltPlayer(fPlayer);
|
|
|
|
} else {
|
|
|
|
faction.addFPlayer(fPlayer);
|
|
|
|
}
|
2019-03-03 04:51:21 +01:00
|
|
|
}
|
2019-05-20 21:25:23 +02:00
|
|
|
|
2019-03-03 04:51:21 +01:00
|
|
|
UtilFly.run();
|
|
|
|
|
|
|
|
Board.getInstance().load();
|
|
|
|
Board.getInstance().clean();
|
|
|
|
|
|
|
|
// Add Base Commands
|
|
|
|
this.cmdBase = new FCmdRoot();
|
|
|
|
this.cmdAutoHelp = new CmdAutoHelp();
|
|
|
|
this.getBaseCommands().add(cmdBase);
|
|
|
|
|
|
|
|
Econ.setup();
|
|
|
|
setupPermissions();
|
|
|
|
|
|
|
|
if (Conf.worldGuardChecking || Conf.worldGuardBuildPriority) {
|
|
|
|
Worldguard.init(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
EngineDynmap.getInstance().init();
|
|
|
|
|
|
|
|
// start up task which runs the autoLeaveAfterDaysOfInactivity routine
|
|
|
|
startAutoLeaveTask(false);
|
|
|
|
|
|
|
|
if (version > 8) {
|
|
|
|
useNonPacketParticles = true;
|
|
|
|
log("Minecraft Version 1.9 or higher found, using non packet based particle API");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (getConfig().getBoolean("enable-faction-flight")) {
|
|
|
|
factionsFlight = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (getServer().getPluginManager().getPlugin("Skript") != null) {
|
2019-08-02 23:01:33 +02:00
|
|
|
log("Skript was found! Registering P Addon...");
|
2019-03-03 04:51:21 +01:00
|
|
|
skriptAddon = Skript.registerAddon(this);
|
|
|
|
try {
|
|
|
|
skriptAddon.loadClasses("com.massivecraft.factions.skript", "expressions");
|
|
|
|
} catch (IOException ex) {
|
|
|
|
ex.printStackTrace();
|
|
|
|
}
|
|
|
|
log("Skript addon registered!");
|
|
|
|
}
|
|
|
|
|
2019-08-04 22:23:40 +02:00
|
|
|
ShopConfig.setup();
|
|
|
|
|
2019-06-04 00:34:10 +02:00
|
|
|
getServer().getPluginManager().registerEvents(factionsPlayerListener = new FactionsPlayerListener(), this);
|
2019-03-03 04:51:21 +01:00
|
|
|
|
|
|
|
// Register Event Handlers
|
|
|
|
eventsListener = new Listener[]{
|
|
|
|
new FactionsChatListener(),
|
|
|
|
new FactionsEntityListener(),
|
|
|
|
new FactionsExploitListener(),
|
|
|
|
new FactionsBlockListener(),
|
|
|
|
new FUpgradesGUI(),
|
|
|
|
new EXPUpgrade(),
|
|
|
|
new CropUpgrades(),
|
2019-05-14 15:27:57 +02:00
|
|
|
new RedstoneUpgrade(),
|
2019-08-04 22:23:40 +02:00
|
|
|
new ShopClickPersistence(),
|
2019-06-02 07:52:34 +02:00
|
|
|
new SpawnerUpgrades()
|
2019-03-03 04:51:21 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
for (Listener eventListener : eventsListener)
|
|
|
|
getServer().getPluginManager().registerEvents(eventListener, this);
|
|
|
|
|
|
|
|
// since some other plugins execute commands directly through this command interface, provide it
|
|
|
|
getCommand(this.refCommand).setExecutor(this);
|
|
|
|
getCommand(this.refCommand).setTabCompleter(this);
|
|
|
|
|
2019-06-02 07:52:34 +02:00
|
|
|
|
2019-08-02 23:01:33 +02:00
|
|
|
RegisteredServiceProvider<Economy> rsp = P.this.getServer().getServicesManager().getRegistration(Economy.class);
|
|
|
|
P.econ = rsp.getProvider();
|
2019-06-02 07:52:34 +02:00
|
|
|
|
2019-03-03 04:51:21 +01:00
|
|
|
if (getDescription().getFullName().contains("BETA")) {
|
|
|
|
divider();
|
|
|
|
System.out.println("You are using a BETA version of the plugin!");
|
|
|
|
System.out.println("This comes with risks of small bugs in newer features!");
|
2019-07-28 11:04:39 +02:00
|
|
|
System.out.println("For support head to: https://github.com/Driftay/Saber-Factions/issues");
|
2019-03-03 04:51:21 +01:00
|
|
|
divider();
|
|
|
|
}
|
|
|
|
|
|
|
|
this.setupPlaceholderAPI();
|
|
|
|
this.postEnable();
|
|
|
|
this.loadSuccessful = true;
|
2019-03-04 22:46:39 +01:00
|
|
|
// Set startup finished to true. to give plugins hooking in a greenlight
|
2019-08-02 23:01:33 +02:00
|
|
|
P.startupFinished = true;
|
2019-03-03 04:51:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public SkriptAddon getSkriptAddon() {
|
|
|
|
return skriptAddon;
|
|
|
|
}
|
|
|
|
|
2019-08-05 14:05:22 +02:00
|
|
|
|
|
|
|
|
2019-03-03 04:51:21 +01:00
|
|
|
private void setupPlaceholderAPI() {
|
|
|
|
Plugin clip = getServer().getPluginManager().getPlugin("PlaceholderAPI");
|
|
|
|
if (clip != null && clip.isEnabled()) {
|
|
|
|
this.clipPlaceholderAPIManager = new ClipPlaceholderAPIManager();
|
|
|
|
if (this.clipPlaceholderAPIManager.register()) {
|
|
|
|
PlaceholderApi = true;
|
|
|
|
log(Level.INFO, "Successfully registered placeholders with PlaceholderAPI.");
|
|
|
|
} else {
|
|
|
|
PlaceholderApi = false;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
PlaceholderApi = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Plugin mvdw = getServer().getPluginManager().getPlugin("MVdWPlaceholderAPI");
|
|
|
|
if (mvdw != null && mvdw.isEnabled()) {
|
|
|
|
this.mvdwPlaceholderAPIManager = true;
|
|
|
|
log(Level.INFO, "Found MVdWPlaceholderAPI. Adding hooks.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-30 08:34:56 +02:00
|
|
|
public List<String> replacePlaceholders(List<String> lore, Placeholder... placeholders) {
|
|
|
|
for (Placeholder placeholder : placeholders) {
|
|
|
|
for (int x = 0; x <= lore.size() - 1; x++) lore.set(x, lore.get(x).replace(placeholder.getTag(), placeholder.getReplace()));
|
|
|
|
}
|
|
|
|
return lore;
|
|
|
|
}
|
|
|
|
|
2019-03-03 04:51:21 +01:00
|
|
|
private void migrateFPlayerLeaders() {
|
|
|
|
List<String> lines = new ArrayList<>();
|
2019-05-15 09:10:10 +02:00
|
|
|
File fplayerFile = new File("plugins" + File.pathSeparator + "Factions" + File.pathSeparator + "players.json");
|
2019-03-03 04:51:21 +01:00
|
|
|
|
|
|
|
try {
|
|
|
|
BufferedReader br = new BufferedReader(new FileReader(fplayerFile));
|
|
|
|
System.out.println("Migrating old players.json file.");
|
|
|
|
|
|
|
|
String line;
|
|
|
|
while ((line = br.readLine()) != null) {
|
|
|
|
if (line.contains("\"role\": \"ADMIN\"")) {
|
|
|
|
line = line.replace("\"role\": \"ADMIN\"", "\"role\": " + "\"LEADER\"");
|
|
|
|
}
|
|
|
|
lines.add(line);
|
|
|
|
}
|
|
|
|
br.close();
|
|
|
|
BufferedWriter bw = new BufferedWriter(new FileWriter(fplayerFile));
|
|
|
|
for (String newLine : lines) {
|
|
|
|
bw.write(newLine + "\n");
|
|
|
|
}
|
|
|
|
bw.flush();
|
|
|
|
bw.close();
|
|
|
|
} catch (IOException ex) {
|
|
|
|
System.out.println("File was not found for players.json, assuming"
|
|
|
|
+ " there is no need to migrate old players.json file.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isClipPlaceholderAPIHooked() {
|
|
|
|
return this.clipPlaceholderAPIManager != null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isMVdWPlaceholderAPIHooked() {
|
|
|
|
return this.mvdwPlaceholderAPIManager;
|
|
|
|
}
|
|
|
|
|
|
|
|
private boolean setupPermissions() {
|
|
|
|
try {
|
|
|
|
RegisteredServiceProvider<Permission> rsp = getServer().getServicesManager().getRegistration(Permission.class);
|
|
|
|
if (rsp != null) {
|
|
|
|
perms = rsp.getProvider();
|
|
|
|
}
|
|
|
|
} catch (NoClassDefFoundError ex) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return perms != null;
|
|
|
|
}
|
|
|
|
|
|
|
|
private boolean setupPlayervaults() {
|
|
|
|
Plugin plugin = getServer().getPluginManager().getPlugin("PlayerVaults");
|
|
|
|
return plugin != null && plugin.isEnabled();
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public GsonBuilder getGsonBuilder() {
|
|
|
|
Type mapFLocToStringSetType = new TypeToken<Map<FLocation, Set<String>>>() {
|
|
|
|
}.getType();
|
|
|
|
|
|
|
|
Type accessTypeAdatper = new TypeToken<Map<Permissable, Map<PermissableAction, Access>>>() {
|
|
|
|
}.getType();
|
|
|
|
|
|
|
|
return new GsonBuilder().setPrettyPrinting().disableHtmlEscaping().enableComplexMapKeySerialization().excludeFieldsWithModifiers(Modifier.TRANSIENT, Modifier.VOLATILE)
|
|
|
|
.registerTypeAdapter(accessTypeAdatper, new PermissionsMapTypeAdapter())
|
|
|
|
.registerTypeAdapter(LazyLocation.class, new MyLocationTypeAdapter())
|
|
|
|
.registerTypeAdapter(mapFLocToStringSetType, new MapFLocToStringSetTypeAdapter())
|
|
|
|
.registerTypeAdapter(Inventory.class, new InventoryTypeAdapter())
|
|
|
|
.registerTypeAdapter(Location.class, new LocationTypeAdapter())
|
|
|
|
.registerTypeAdapterFactory(EnumTypeAdapter.ENUM_FACTORY);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void divider() {
|
|
|
|
System.out.println(" .-.-. .-.-. .-.-. .-.-. .-.-. .-.-. .-.-. .-.-");
|
|
|
|
System.out.println(" / / \\ \\ / / \\ \\ / / \\ \\ / / \\ \\ / / \\ \\ / / \\ \\ / / \\ \\ / / \\");
|
|
|
|
System.out.println("`-' `-`-' `-`-' `-`-' `-`-' `-`-' `-`-' `-`-'");
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onDisable() {
|
|
|
|
// only save data if plugin actually completely loaded successfully
|
|
|
|
if (this.loadSuccessful) {
|
|
|
|
// Dont save, as this is kind of pointless, as the /f config command manually saves.
|
|
|
|
// So any edits done are saved, this way manual edits to json can go through.
|
|
|
|
|
|
|
|
// Conf.save();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (AutoLeaveTask != null) {
|
|
|
|
this.getServer().getScheduler().cancelTask(AutoLeaveTask);
|
|
|
|
AutoLeaveTask = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
super.onDisable();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void startAutoLeaveTask(boolean restartIfRunning) {
|
|
|
|
if (AutoLeaveTask != null) {
|
|
|
|
if (!restartIfRunning) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this.getServer().getScheduler().cancelTask(AutoLeaveTask);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Conf.autoLeaveRoutineRunsEveryXMinutes > 0.0) {
|
|
|
|
long ticks = (long) (20 * 60 * Conf.autoLeaveRoutineRunsEveryXMinutes);
|
|
|
|
AutoLeaveTask = getServer().getScheduler().scheduleSyncRepeatingTask(this, new AutoLeaveTask(), ticks, ticks);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void postAutoSave() {
|
|
|
|
//Board.getInstance().forceSave(); Not sure why this was there as it's called after the board is already saved.
|
|
|
|
Conf.save();
|
|
|
|
}
|
|
|
|
|
|
|
|
public ItemStack createItem(Material material, int amount, short datavalue, String name, List<String> lore) {
|
2019-07-17 17:08:55 +02:00
|
|
|
ItemStack item = new ItemStack(XMaterial.matchXMaterial(material.toString()).parseMaterial(), amount, datavalue);
|
2019-03-03 04:51:21 +01:00
|
|
|
ItemMeta meta = item.getItemMeta();
|
|
|
|
meta.setDisplayName(color(name));
|
|
|
|
meta.setLore(colorList(lore));
|
|
|
|
item.setItemMeta(meta);
|
|
|
|
return item;
|
|
|
|
}
|
|
|
|
|
|
|
|
public ItemStack createLazyItem(Material material, int amount, short datavalue, String name, String lore) {
|
|
|
|
ItemStack item = new ItemStack(material, amount, datavalue);
|
|
|
|
ItemMeta meta = item.getItemMeta();
|
2019-08-02 23:01:33 +02:00
|
|
|
meta.setDisplayName(color(P.p.getConfig().getString(name)));
|
|
|
|
meta.setLore(colorList(P.p.getConfig().getStringList(lore)));
|
2019-03-03 04:51:21 +01:00
|
|
|
item.setItemMeta(meta);
|
|
|
|
return item;
|
|
|
|
}
|
|
|
|
|
|
|
|
public Economy getEcon() {
|
2019-08-02 23:01:33 +02:00
|
|
|
RegisteredServiceProvider<Economy> rsp = P.p.getServer().getServicesManager().getRegistration(Economy.class);
|
2019-03-03 04:51:21 +01:00
|
|
|
Economy econ = rsp.getProvider();
|
|
|
|
return econ;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public boolean logPlayerCommands() {
|
|
|
|
return Conf.logPlayerCommands;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public boolean handleCommand(CommandSender sender, String commandString, boolean testOnly) {
|
|
|
|
return sender instanceof Player && FactionsPlayerListener.preventCommand(commandString, (Player) sender) || super.handleCommand(sender, commandString, testOnly);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public boolean onCommand(CommandSender sender, Command command, String label, String[] split) {
|
|
|
|
if (split.length == 0) {
|
|
|
|
return handleCommand(sender, "/f help", false);
|
|
|
|
}
|
|
|
|
|
|
|
|
// otherwise, needs to be handled; presumably another plugin directly ran the command
|
|
|
|
String cmd = Conf.baseCommandAliases.isEmpty() ? "/f" : "/" + Conf.baseCommandAliases.get(0);
|
|
|
|
return handleCommand(sender, cmd + " " + TextUtil.implode(Arrays.asList(split), " "), false);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public List<String> onTabComplete(CommandSender sender, Command command, String alias, String[] args) {
|
|
|
|
FPlayer fPlayer = FPlayers.getInstance().getByPlayer((Player) sender);
|
|
|
|
List<String> completions = new ArrayList<>();
|
|
|
|
String cmd = Conf.baseCommandAliases.isEmpty() ? "/f" : "/" + Conf.baseCommandAliases.get(0);
|
|
|
|
List<String> argsList = new ArrayList<>(Arrays.asList(args));
|
|
|
|
argsList.remove(argsList.size() - 1);
|
|
|
|
String cmdValid = (cmd + " " + TextUtil.implode(argsList, " ")).trim();
|
|
|
|
MCommand<?> commandEx = cmdBase;
|
|
|
|
List<MCommand<?>> commandsList = cmdBase.subCommands;
|
|
|
|
|
|
|
|
for (; !commandsList.isEmpty() && !argsList.isEmpty(); argsList.remove(0)) {
|
|
|
|
String cmdName = argsList.get(0).toLowerCase();
|
|
|
|
MCommand<?> commandFounded = commandsList.stream()
|
|
|
|
.filter(c -> c.aliases.contains(cmdName))
|
|
|
|
.findFirst().orElse(null);
|
|
|
|
|
|
|
|
if (commandFounded != null) {
|
|
|
|
commandEx = commandFounded;
|
|
|
|
commandsList = commandFounded.subCommands;
|
|
|
|
} else break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (argsList.isEmpty()) {
|
|
|
|
for (MCommand<?> subCommand : commandEx.subCommands) {
|
|
|
|
subCommand.setCommandSender(sender);
|
|
|
|
if (handleCommand(sender, cmdValid + " " + subCommand.aliases.get(0), true)
|
|
|
|
&& subCommand.visibility != CommandVisibility.INVISIBLE
|
|
|
|
&& subCommand.validSenderType(sender, false)
|
|
|
|
&& subCommand.validSenderPermissions(sender, false))
|
|
|
|
completions.addAll(subCommand.aliases);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
String lastArg = args[args.length - 1].toLowerCase();
|
2019-04-15 20:09:35 +02:00
|
|
|
for (Player player : Bukkit.getServer().getOnlinePlayers()) {
|
|
|
|
completions.add(player.getName());
|
|
|
|
}
|
2019-07-17 17:08:55 +02:00
|
|
|
completions = completions.stream()
|
|
|
|
.filter(m -> m.toLowerCase().startsWith(lastArg))
|
|
|
|
.collect(Collectors.toList());
|
2019-04-15 20:09:35 +02:00
|
|
|
return completions;
|
2019-03-03 04:51:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public void createTimedHologram(final Location location, String text, Long timeout) {
|
|
|
|
ArmorStand as = (ArmorStand) location.add(0.5, 1, 0.5).getWorld().spawnEntity(location, EntityType.ARMOR_STAND); //Spawn the ArmorStand
|
|
|
|
as.setVisible(false); //Makes the ArmorStand invisible
|
|
|
|
as.setGravity(false); //Make sure it doesn't fall
|
|
|
|
as.setCanPickupItems(false); //I'm not sure what happens if you leave this as it is, but you might as well disable it
|
2019-08-02 23:01:33 +02:00
|
|
|
as.setCustomName(P.p.color(text)); //Set this to the text you want
|
2019-03-03 04:51:21 +01:00
|
|
|
as.setCustomNameVisible(true); //This makes the text appear no matter if your looking at the entity or not
|
|
|
|
final ArmorStand armorStand = as;
|
|
|
|
|
2019-08-02 23:01:33 +02:00
|
|
|
Bukkit.getScheduler().scheduleSyncDelayedTask(P.p, () -> {
|
2019-03-03 04:51:21 +01:00
|
|
|
armorStand.remove();
|
|
|
|
getLogger().info("Removing Hologram.");
|
|
|
|
}
|
|
|
|
, timeout * 20);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// -------------------------------------------- //
|
|
|
|
// Functions for other plugins to hook into
|
|
|
|
// -------------------------------------------- //
|
|
|
|
|
|
|
|
// This value will be updated whenever new hooks are added
|
|
|
|
public int hookSupportVersion() {
|
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If another plugin is handling insertion of chat tags, this should be used to notify Factions
|
|
|
|
public void handleFactionTagExternally(boolean notByFactions) {
|
|
|
|
Conf.chatTagHandledByAnotherPlugin = notByFactions;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Simply put, should this chat event be left for Factions to handle? For now, that means players with Faction Chat
|
|
|
|
// enabled or use of the Factions f command without a slash; combination of isPlayerFactionChatting() and isFactionsCommand()
|
|
|
|
|
|
|
|
public boolean shouldLetFactionsHandleThisChat(AsyncPlayerChatEvent event) {
|
|
|
|
return event != null && (isPlayerFactionChatting(event.getPlayer()) || isFactionsCommand(event.getMessage()));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Does player have Faction Chat enabled? If so, chat plugins should preferably not do channels,
|
|
|
|
// local chat, or anything else which targets individual recipients, so Faction Chat can be done
|
|
|
|
public boolean isPlayerFactionChatting(Player player) {
|
|
|
|
if (player == null) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
FPlayer me = FPlayers.getInstance().getByPlayer(player);
|
|
|
|
|
|
|
|
return me != null && me.getChatMode().isAtLeast(ChatMode.ALLIANCE);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Is this chat message actually a Factions command, and thus should be left alone by other plugins?
|
|
|
|
|
|
|
|
// TODO: GET THIS BACK AND WORKING
|
|
|
|
|
|
|
|
public boolean isFactionsCommand(String check) {
|
|
|
|
return !(check == null || check.isEmpty()) && this.handleCommand(null, check, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get a player's faction tag (faction name), mainly for usage by chat plugins for local/channel chat
|
|
|
|
public String getPlayerFactionTag(Player player) {
|
|
|
|
return getPlayerFactionTagRelation(player, null);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Same as above, but with relation (enemy/neutral/ally) coloring potentially added to the tag
|
|
|
|
public String getPlayerFactionTagRelation(Player speaker, Player listener) {
|
|
|
|
String tag = "~";
|
|
|
|
|
|
|
|
if (speaker == null) {
|
|
|
|
return tag;
|
|
|
|
}
|
|
|
|
|
|
|
|
FPlayer me = FPlayers.getInstance().getByPlayer(speaker);
|
|
|
|
if (me == null) {
|
|
|
|
return tag;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if listener isn't set, or config option is disabled, give back uncolored tag
|
|
|
|
if (listener == null || !Conf.chatTagRelationColored) {
|
|
|
|
tag = me.getChatTag().trim();
|
|
|
|
} else {
|
|
|
|
FPlayer you = FPlayers.getInstance().getByPlayer(listener);
|
|
|
|
if (you == null) {
|
|
|
|
tag = me.getChatTag().trim();
|
|
|
|
} else // everything checks out, give the colored tag
|
|
|
|
{
|
|
|
|
tag = me.getChatTag(you).trim();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (tag.isEmpty()) {
|
|
|
|
tag = "~";
|
|
|
|
}
|
|
|
|
|
|
|
|
return tag;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get a player's title within their faction, mainly for usage by chat plugins for local/channel chat
|
|
|
|
public String getPlayerTitle(Player player) {
|
|
|
|
if (player == null) {
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
FPlayer me = FPlayers.getInstance().getByPlayer(player);
|
|
|
|
if (me == null) {
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
return me.getTitle().trim();
|
|
|
|
}
|
|
|
|
|
|
|
|
public String color(String line) {
|
|
|
|
line = ChatColor.translateAlternateColorCodes('&', line);
|
|
|
|
return line;
|
|
|
|
}
|
|
|
|
|
|
|
|
//colors a string list
|
|
|
|
public List<String> colorList(List<String> lore) {
|
|
|
|
for (int i = 0; i <= lore.size() - 1; i++) {
|
|
|
|
lore.set(i, color(lore.get(i)));
|
|
|
|
}
|
|
|
|
return lore;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get a list of all faction tags (names)
|
|
|
|
public Set<String> getFactionTags() {
|
|
|
|
return Factions.getInstance().getFactionTags();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get a list of all players in the specified faction
|
|
|
|
public Set<String> getPlayersInFaction(String factionTag) {
|
|
|
|
Set<String> players = new HashSet<>();
|
|
|
|
Faction faction = Factions.getInstance().getByTag(factionTag);
|
|
|
|
if (faction != null) {
|
|
|
|
for (FPlayer fplayer : faction.getFPlayers()) {
|
|
|
|
players.add(fplayer.getName());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return players;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get a list of all online players in the specified faction
|
|
|
|
public Set<String> getOnlinePlayersInFaction(String factionTag) {
|
|
|
|
Set<String> players = new HashSet<>();
|
|
|
|
Faction faction = Factions.getInstance().getByTag(factionTag);
|
|
|
|
if (faction != null) {
|
|
|
|
for (FPlayer fplayer : faction.getFPlayersWhereOnline(true)) {
|
|
|
|
players.add(fplayer.getName());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return players;
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isHookedPlayervaults() {
|
|
|
|
return hookedPlayervaults;
|
|
|
|
}
|
|
|
|
|
|
|
|
public String getPrimaryGroup(OfflinePlayer player) {
|
2019-05-15 09:10:10 +02:00
|
|
|
AtomicReference<String> primaryGroup = new AtomicReference<>();
|
|
|
|
|
|
|
|
if (perms == null || !perms.hasGroupSupport()) return " ";
|
|
|
|
else {
|
|
|
|
Bukkit.getScheduler().runTaskAsynchronously(this, () -> primaryGroup.set(perms.getPrimaryGroup(Bukkit.getWorlds().get(0).toString(), player)));
|
|
|
|
return primaryGroup.get();
|
|
|
|
}
|
2019-03-03 04:51:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public void debug(Level level, String s) {
|
|
|
|
if (getConfig().getBoolean("debug", false)) {
|
|
|
|
getLogger().log(level, s);
|
|
|
|
}
|
|
|
|
}
|
2019-06-04 00:34:10 +02:00
|
|
|
public FactionsPlayerListener getFactionsPlayerListener() {
|
|
|
|
return this.factionsPlayerListener;
|
|
|
|
}
|
2019-03-03 04:51:21 +01:00
|
|
|
|
|
|
|
public void debug(String s) {
|
|
|
|
debug(Level.INFO, s);
|
|
|
|
}
|
2011-10-08 22:03:44 +02:00
|
|
|
}
|