Fix #412, auto team balance for autoassign warzones

This commit is contained in:
Connor Monahan 2017-07-29 01:26:30 -05:00
parent eb7b600aef
commit 22cbba62b3

View File

@ -1,37 +1,34 @@
package com.tommytony.war; package com.tommytony.war;
import java.sql.Connection; import com.google.common.collect.ImmutableList;
import java.sql.PreparedStatement; import com.tommytony.war.config.*;
import java.sql.ResultSet; import com.tommytony.war.event.WarBattleWinEvent;
import java.sql.SQLException; import com.tommytony.war.event.WarPlayerLeaveEvent;
import java.text.MessageFormat; import com.tommytony.war.event.WarPlayerThiefEvent;
import java.util.*; import com.tommytony.war.event.WarScoreCapEvent;
import java.util.logging.Level; import com.tommytony.war.job.InitZoneJob;
import com.tommytony.war.job.LoadoutResetJob;
import com.tommytony.war.job.LogKillsDeathsJob;
import com.tommytony.war.job.LogKillsDeathsJob.KillsDeathsRecord;
import com.tommytony.war.job.ZoneTimeJob; import com.tommytony.war.job.ZoneTimeJob;
import com.tommytony.war.mapper.LoadoutYmlMapper;
import com.tommytony.war.mapper.VolumeMapper;
import com.tommytony.war.mapper.ZoneVolumeMapper;
import com.tommytony.war.spout.SpoutDisplayer;
import com.tommytony.war.structure.*; import com.tommytony.war.structure.*;
import com.tommytony.war.utility.*;
import com.tommytony.war.volume.Volume;
import com.tommytony.war.volume.ZoneVolume;
import net.milkbowl.vault.economy.EconomyResponse; import net.milkbowl.vault.economy.EconomyResponse;
import org.apache.commons.lang.StringUtils; import org.apache.commons.lang.StringUtils;
import org.apache.commons.lang.Validate; import org.apache.commons.lang.Validate;
import org.bukkit.Bukkit; import org.bukkit.*;
import org.bukkit.ChatColor;
import org.bukkit.GameMode;
import org.bukkit.Location;
import org.bukkit.Material;
import org.bukkit.OfflinePlayer;
import org.bukkit.World;
import org.bukkit.attribute.Attribute; import org.bukkit.attribute.Attribute;
import org.bukkit.attribute.AttributeInstance; import org.bukkit.attribute.AttributeInstance;
import org.bukkit.attribute.AttributeModifier; import org.bukkit.attribute.AttributeModifier;
import org.bukkit.block.Block; import org.bukkit.block.Block;
import org.bukkit.block.BlockFace; import org.bukkit.block.BlockFace;
import org.bukkit.entity.Arrow; import org.bukkit.entity.*;
import org.bukkit.entity.Entity;
import org.bukkit.entity.Item;
import org.bukkit.entity.Player;
import org.bukkit.entity.Projectile;
import org.bukkit.entity.TNTPrimed;
import org.bukkit.event.entity.EntityDamageByEntityEvent; import org.bukkit.event.entity.EntityDamageByEntityEvent;
import org.bukkit.event.entity.EntityDamageEvent; import org.bukkit.event.entity.EntityDamageEvent;
import org.bukkit.event.entity.EntityDamageEvent.DamageCause; import org.bukkit.event.entity.EntityDamageEvent.DamageCause;
@ -50,33 +47,13 @@ import org.bukkit.scoreboard.Scoreboard;
import org.getspout.spoutapi.SpoutManager; import org.getspout.spoutapi.SpoutManager;
import org.getspout.spoutapi.player.SpoutPlayer; import org.getspout.spoutapi.player.SpoutPlayer;
import com.google.common.collect.ImmutableList; import java.sql.Connection;
import com.tommytony.war.config.InventoryBag; import java.sql.PreparedStatement;
import com.tommytony.war.config.ScoreboardType; import java.sql.ResultSet;
import com.tommytony.war.config.TeamConfig; import java.sql.SQLException;
import com.tommytony.war.config.TeamConfigBag; import java.text.MessageFormat;
import com.tommytony.war.config.TeamKind; import java.util.*;
import com.tommytony.war.config.WarzoneConfig; import java.util.logging.Level;
import com.tommytony.war.config.WarzoneConfigBag;
import com.tommytony.war.event.WarBattleWinEvent;
import com.tommytony.war.event.WarPlayerLeaveEvent;
import com.tommytony.war.event.WarPlayerThiefEvent;
import com.tommytony.war.event.WarScoreCapEvent;
import com.tommytony.war.job.InitZoneJob;
import com.tommytony.war.job.LoadoutResetJob;
import com.tommytony.war.job.LogKillsDeathsJob;
import com.tommytony.war.job.LogKillsDeathsJob.KillsDeathsRecord;
import com.tommytony.war.mapper.LoadoutYmlMapper;
import com.tommytony.war.mapper.VolumeMapper;
import com.tommytony.war.mapper.ZoneVolumeMapper;
import com.tommytony.war.spout.SpoutDisplayer;
import com.tommytony.war.utility.Direction;
import com.tommytony.war.utility.Loadout;
import com.tommytony.war.utility.LoadoutSelection;
import com.tommytony.war.utility.PlayerState;
import com.tommytony.war.utility.PotionEffectHelper;
import com.tommytony.war.volume.Volume;
import com.tommytony.war.volume.ZoneVolume;
/** /**
* *
@ -86,28 +63,29 @@ import com.tommytony.war.volume.ZoneVolume;
public class Warzone { public class Warzone {
public enum LeaveCause { static final Comparator<Team> LEAST_PLAYER_COUNT_ORDER = new Comparator<Team>() {
COMMAND, DISCONNECT, SCORECAP, RESET; @Override
public boolean useRallyPoint() { public int compare(Team arg0, Team arg1) {
return this == SCORECAP ? true : false; return arg0.getPlayers().size() - arg1.getPlayers().size();
} }
} };
private String name;
private ZoneVolume volume;
private World world;
private final List<Team> teams = new ArrayList<Team>(); private final List<Team> teams = new ArrayList<Team>();
private final List<Monument> monuments = new ArrayList<Monument>(); private final List<Monument> monuments = new ArrayList<Monument>();
private final List<CapturePoint> capturePoints = new ArrayList<CapturePoint>(); private final List<CapturePoint> capturePoints = new ArrayList<CapturePoint>();
private final List<Bomb> bombs = new ArrayList<Bomb>(); private final List<Bomb> bombs = new ArrayList<Bomb>();
private final List<Cake> cakes = new ArrayList<Cake>(); private final List<Cake> cakes = new ArrayList<Cake>();
private final List<String> authors = new ArrayList<String>();
private final int minSafeDistanceFromWall = 6;
private final List<Player> respawn = new ArrayList<Player>();
private final List<String> reallyDeadFighters = new ArrayList<String>();
private final WarzoneConfigBag warzoneConfig;
private final TeamConfigBag teamDefaultConfig;
private String name;
private ZoneVolume volume;
private World world;
private Location teleport; private Location teleport;
private ZoneLobby lobby; private ZoneLobby lobby;
private Location rallyPoint; private Location rallyPoint;
private final List<String> authors = new ArrayList<String>();
private final int minSafeDistanceFromWall = 6;
private List<ZoneWallGuard> zoneWallGuards = new ArrayList<ZoneWallGuard>(); private List<ZoneWallGuard> zoneWallGuards = new ArrayList<ZoneWallGuard>();
private HashMap<String, PlayerState> playerStates = new HashMap<String, PlayerState>(); private HashMap<String, PlayerState> playerStates = new HashMap<String, PlayerState>();
private HashMap<UUID, Team> flagThieves = new HashMap<UUID, Team>(); private HashMap<UUID, Team> flagThieves = new HashMap<UUID, Team>();
@ -116,15 +94,9 @@ public class Warzone {
private HashMap<String, LoadoutSelection> loadoutSelections = new HashMap<String, LoadoutSelection>(); private HashMap<String, LoadoutSelection> loadoutSelections = new HashMap<String, LoadoutSelection>();
private HashMap<String, PlayerState> deadMenInventories = new HashMap<String, PlayerState>(); private HashMap<String, PlayerState> deadMenInventories = new HashMap<String, PlayerState>();
private HashMap<String, Integer> killCount = new HashMap<String, Integer>(); private HashMap<String, Integer> killCount = new HashMap<String, Integer>();
private final List<Player> respawn = new ArrayList<Player>();
private final List<String> reallyDeadFighters = new ArrayList<String>();
private HashMap<Player, PermissionAttachment> attachments = new HashMap<Player, PermissionAttachment>(); private HashMap<Player, PermissionAttachment> attachments = new HashMap<Player, PermissionAttachment>();
private HashMap<Player, Team> delayedJoinPlayers = new HashMap<Player, Team>(); private HashMap<Player, Team> delayedJoinPlayers = new HashMap<Player, Team>();
private List<LogKillsDeathsJob.KillsDeathsRecord> killsDeathsTracker = new ArrayList<KillsDeathsRecord>(); private List<LogKillsDeathsJob.KillsDeathsRecord> killsDeathsTracker = new ArrayList<KillsDeathsRecord>();
private final WarzoneConfigBag warzoneConfig;
private final TeamConfigBag teamDefaultConfig;
private InventoryBag defaultInventories = new InventoryBag(); private InventoryBag defaultInventories = new InventoryBag();
private Scoreboard scoreboard; private Scoreboard scoreboard;
@ -139,7 +111,10 @@ public class Warzone {
//private final Object gameEndLock = new Object(); //private final Object gameEndLock = new Object();
private boolean pvpReady = true; private boolean pvpReady = true;
private Random killSeed = new Random();
// prevent tryCallDelayedPlayers from being recursively called by Warzone#assign
private boolean activeDelayedCall = false;
public Warzone(World world, String name) { public Warzone(World world, String name) {
this.world = world; this.world = world;
this.name = name; this.name = name;
@ -204,10 +179,7 @@ public class Warzone {
} }
public boolean ready() { public boolean ready() {
if (this.volume.hasTwoCorners() && !this.volume.tooSmall() && !this.volume.tooBig()) { return this.volume.hasTwoCorners() && !this.volume.tooSmall() && !this.volume.tooBig();
return true;
}
return false;
} }
public List<Team> getTeams() { public List<Team> getTeams() {
@ -244,19 +216,24 @@ public class Warzone {
return this.name; return this.name;
} }
public void setName(String newName) {
this.name = newName;
this.volume.setName(newName);
}
@Override @Override
public String toString() { public String toString() {
return this.getName(); return this.getName();
} }
public void setTeleport(Location location) {
this.teleport = location;
}
public Location getTeleport() { public Location getTeleport() {
return this.teleport; return this.teleport;
} }
public void setTeleport(Location location) {
this.teleport = location;
}
public int saveState(boolean clearArtifacts) { public int saveState(boolean clearArtifacts) {
if (this.ready()) { if (this.ready()) {
if (clearArtifacts) { if (clearArtifacts) {
@ -350,12 +327,12 @@ public class Warzone {
} }
this.initZone(); this.initZone();
if (War.war.getWarHub() != null) { if (War.war.getWarHub() != null) {
War.war.getWarHub().resetZoneSign(this); War.war.getWarHub().resetZoneSign(this);
} }
} }
// Don't forget to reset these to false, or we won't be able to score or empty lifepools anymore // Don't forget to reset these to false, or we won't be able to score or empty lifepools anymore
this.isReinitializing = false; this.isReinitializing = false;
this.isEndOfGame = false; this.isEndOfGame = false;
@ -383,13 +360,13 @@ public class Warzone {
cp.getVolume().resetBlocks(); cp.getVolume().resetBlocks();
cp.reset(); cp.reset();
} }
// reset bombs // reset bombs
for (Bomb bomb : this.bombs) { for (Bomb bomb : this.bombs) {
bomb.getVolume().resetBlocks(); bomb.getVolume().resetBlocks();
bomb.addBombBlocks(); bomb.addBombBlocks();
} }
// reset cakes // reset cakes
for (Cake cake : this.cakes) { for (Cake cake : this.cakes) {
cake.getVolume().resetBlocks(); cake.getVolume().resetBlocks();
@ -428,10 +405,10 @@ public class Warzone {
} }
} }
this.reallyDeadFighters.clear(); this.reallyDeadFighters.clear();
//get them config (here be crazy grinning's!) //get them config (here be crazy grinning's!)
int pvpready = warzoneConfig.getInt(WarzoneConfig.PREPTIME); int pvpready = warzoneConfig.getInt(WarzoneConfig.PREPTIME);
if(pvpready != 0) { //if it is equalz to zeroz then dinosaurs will take over the earth if(pvpready != 0) { //if it is equalz to zeroz then dinosaurs will take over the earth
this.pvpReady = false; this.pvpReady = false;
ZoneTimeJob timer = new ZoneTimeJob(this); ZoneTimeJob timer = new ZoneTimeJob(this);
@ -443,7 +420,7 @@ public class Warzone {
if (!(entity instanceof Item)) { if (!(entity instanceof Item)) {
continue; continue;
} }
// validate position // validate position
if (!this.getVolume().contains(entity.getLocation())) { if (!this.getVolume().contains(entity.getLocation())) {
continue; continue;
@ -469,7 +446,7 @@ public class Warzone {
// Teleport the player back to spawn // Teleport the player back to spawn
event.setTo(team.getRandomSpawn()); event.setTo(team.getRandomSpawn());
} }
public boolean isRespawning(Player p) { public boolean isRespawning(Player p) {
return respawn.contains(p); return respawn.contains(p);
} }
@ -503,7 +480,7 @@ public class Warzone {
// Stop fire here, since doing it in the same tick as death doesn't extinguish it // Stop fire here, since doing it in the same tick as death doesn't extinguish it
player.setFireTicks(0); player.setFireTicks(0);
} }
}; };
// ughhhhh bukkit // ughhhhh bukkit
War.war.getServer().getScheduler().runTaskLater(War.war, antiFireAction, 1L); War.war.getServer().getScheduler().runTaskLater(War.war, antiFireAction, 1L);
@ -547,7 +524,7 @@ public class Warzone {
player.addPotionEffect(new PotionEffect(PotionEffectType.BLINDNESS, respawnTimeTicks, 255)); player.addPotionEffect(new PotionEffect(PotionEffectType.BLINDNESS, respawnTimeTicks, 255));
player.sendTitle("", ChatColor.RED + MessageFormat.format(War.war.getString("zone.spawn.timer.title"), respawnTime), 1, respawnTimeTicks, 10); player.sendTitle("", ChatColor.RED + MessageFormat.format(War.war.getString("zone.spawn.timer.title"), respawnTime), 1, respawnTimeTicks, 10);
} }
boolean isFirstRespawn = false; boolean isFirstRespawn = false;
if (!this.getLoadoutSelections().keySet().contains(player.getName())) { if (!this.getLoadoutSelections().keySet().contains(player.getName())) {
isFirstRespawn = true; isFirstRespawn = true;
@ -558,7 +535,7 @@ public class Warzone {
} else { } else {
this.getLoadoutSelections().get(player.getName()).setStillInSpawn(true); this.getLoadoutSelections().get(player.getName()).setStillInSpawn(true);
} }
// Spout // Spout
if (War.war.isSpoutServer()) { if (War.war.isSpoutServer()) {
SpoutManager.getPlayer(player).setTitle(team.getKind().getColor() + player.getName()); SpoutManager.getPlayer(player).setTitle(team.getKind().getColor() + player.getName());
@ -569,11 +546,11 @@ public class Warzone {
final LoadoutResetJob job = new LoadoutResetJob(this, team, player, isFirstRespawn, false); final LoadoutResetJob job = new LoadoutResetJob(this, team, player, isFirstRespawn, false);
if (team.getTeamConfig().resolveInt(TeamConfig.RESPAWNTIMER) == 0 || isFirstRespawn) { if (team.getTeamConfig().resolveInt(TeamConfig.RESPAWNTIMER) == 0 || isFirstRespawn) {
job.run(); job.run();
} }
else { else {
// "Respawn" Timer - player will not be able to leave spawn for a few seconds // "Respawn" Timer - player will not be able to leave spawn for a few seconds
respawn.add(player); respawn.add(player);
War.war.getServer().getScheduler().scheduleSyncDelayedTask(War.war, new Runnable() { War.war.getServer().getScheduler().scheduleSyncDelayedTask(War.war, new Runnable() {
public void run() { public void run() {
respawn.remove(player); respawn.remove(player);
@ -669,12 +646,12 @@ public class Warzone {
public void keepPlayerState(Player player) { public void keepPlayerState(Player player) {
PlayerInventory inventory = player.getInventory(); PlayerInventory inventory = player.getInventory();
ItemStack[] contents = inventory.getContents(); ItemStack[] contents = inventory.getContents();
String playerTitle = player.getName(); String playerTitle = player.getName();
if (War.war.isSpoutServer()) { if (War.war.isSpoutServer()) {
playerTitle = SpoutManager.getPlayer(player).getTitle(); playerTitle = SpoutManager.getPlayer(player).getTitle();
} }
this.playerStates.put( this.playerStates.put(
player.getName(), player.getName(),
new PlayerState(player.getGameMode(), contents, inventory new PlayerState(player.getGameMode(), contents, inventory
@ -692,7 +669,7 @@ public class Warzone {
if (originalState != null) { if (originalState != null) {
// prevent item hacking thru CRAFTING personal inventory slots // prevent item hacking thru CRAFTING personal inventory slots
this.preventItemHackingThroughOpenedInventory(player); this.preventItemHackingThroughOpenedInventory(player);
this.playerInvFromInventoryStash(playerInv, originalState); this.playerInvFromInventoryStash(playerInv, originalState);
player.setGameMode(originalState.getGamemode()); player.setGameMode(originalState.getGamemode());
double maxH = player.getAttribute(Attribute.GENERIC_MAX_HEALTH).getValue(); double maxH = player.getAttribute(Attribute.GENERIC_MAX_HEALTH).getValue();
@ -704,7 +681,7 @@ public class Warzone {
player.setLevel(originalState.getLevel()); player.setLevel(originalState.getLevel());
player.setExp(originalState.getExp()); player.setExp(originalState.getExp());
player.setAllowFlight(originalState.canFly()); player.setAllowFlight(originalState.canFly());
if (War.war.isSpoutServer()) { if (War.war.isSpoutServer()) {
SpoutManager.getPlayer(player).setTitle(originalState.getPlayerTitle()); SpoutManager.getPlayer(player).setTitle(originalState.getPlayerTitle());
} }
@ -715,23 +692,23 @@ public class Warzone {
private void preventItemHackingThroughOpenedInventory(Player player) { private void preventItemHackingThroughOpenedInventory(Player player) {
InventoryView openedInv = player.getOpenInventory(); InventoryView openedInv = player.getOpenInventory();
if (openedInv.getType() == InventoryType.CRAFTING) { if (openedInv.getType() == InventoryType.CRAFTING) {
// prevent abuse of personal crafting slots (this behavior doesn't seem to happen // prevent abuse of personal crafting slots (this behavior doesn't seem to happen
// for containers like workbench and furnace - those get closed properly) // for containers like workbench and furnace - those get closed properly)
openedInv.getTopInventory().clear(); openedInv.getTopInventory().clear();
} }
// Prevent player from keeping items he was transferring in his inventory // Prevent player from keeping items he was transferring in his inventory
openedInv.setCursor(null); openedInv.setCursor(null);
} }
private void playerInvFromInventoryStash(PlayerInventory playerInv, PlayerState originalContents) { private void playerInvFromInventoryStash(PlayerInventory playerInv, PlayerState originalContents) {
playerInv.clear(); playerInv.clear();
playerInv.clear(playerInv.getSize() + 0); playerInv.clear(playerInv.getSize() + 0);
playerInv.clear(playerInv.getSize() + 1); playerInv.clear(playerInv.getSize() + 1);
playerInv.clear(playerInv.getSize() + 2); playerInv.clear(playerInv.getSize() + 2);
playerInv.clear(playerInv.getSize() + 3); // helmet/blockHead playerInv.clear(playerInv.getSize() + 3); // helmet/blockHead
int invIndex = 0; int invIndex = 0;
for (ItemStack item : originalContents.getContents()) { for (ItemStack item : originalContents.getContents()) {
if (item != null && item.getType() != Material.AIR) { if (item != null && item.getType() != Material.AIR) {
@ -739,7 +716,7 @@ public class Warzone {
} }
invIndex++; invIndex++;
} }
if (originalContents.getHelmet() != null) { if (originalContents.getHelmet() != null) {
playerInv.setHelmet(originalContents.getHelmet()); playerInv.setHelmet(originalContents.getHelmet());
} }
@ -784,7 +761,7 @@ public class Warzone {
} }
return null; return null;
} }
public boolean hasBomb(String bombName) { public boolean hasBomb(String bombName) {
for (Bomb bomb : this.bombs) { for (Bomb bomb : this.bombs) {
if (bomb.getName().equals(bombName)) { if (bomb.getName().equals(bombName)) {
@ -802,7 +779,7 @@ public class Warzone {
} }
return null; return null;
} }
public boolean hasCake(String cakeName) { public boolean hasCake(String cakeName) {
for (Cake cake : this.cakes) { for (Cake cake : this.cakes) {
if (cake.getName().equals(cakeName)) { if (cake.getName().equals(cakeName)) {
@ -1025,21 +1002,14 @@ public class Warzone {
playerGuards.clear(); playerGuards.clear();
} }
public void setLobby(ZoneLobby lobby) {
this.lobby = lobby;
}
public ZoneLobby getLobby() { public ZoneLobby getLobby() {
return this.lobby; return this.lobby;
} }
static final Comparator<Team> LEAST_PLAYER_COUNT_ORDER = new Comparator<Team>() { public void setLobby(ZoneLobby lobby) {
@Override this.lobby = lobby;
public int compare(Team arg0, Team arg1) { }
return arg0.getPlayers().size() - arg1.getPlayers().size();
}
};
public Team autoAssign(Player player) { public Team autoAssign(Player player) {
Collections.sort(teams, LEAST_PLAYER_COUNT_ORDER); Collections.sort(teams, LEAST_PLAYER_COUNT_ORDER);
Team lowestNoOfPlayers = null; Team lowestNoOfPlayers = null;
@ -1057,7 +1027,7 @@ public class Warzone {
/** /**
* Assign a player to a specific team. * Assign a player to a specific team.
* *
* @param player * @param player
* Player to assign to team. * Player to assign to team.
* @param team * @param team
@ -1101,8 +1071,6 @@ public class Warzone {
} }
} }
private Random killSeed = new Random();
/** /**
* Send death messages and process other records before passing off the * Send death messages and process other records before passing off the
* death to the {@link #handleDeath(Player)} method. * death to the {@link #handleDeath(Player)} method.
@ -1173,7 +1141,7 @@ public class Warzone {
} }
this.handleDeath(player); this.handleDeath(player);
} }
/** /**
* Handle a player killed naturally (like by a dispenser or explosion). * Handle a player killed naturally (like by a dispenser or explosion).
* @param player Player killed * @param player Player killed
@ -1185,7 +1153,7 @@ public class Warzone {
if (event instanceof EntityDamageByEntityEvent if (event instanceof EntityDamageByEntityEvent
&& ((EntityDamageByEntityEvent) event).getDamager() instanceof TNTPrimed) { && ((EntityDamageByEntityEvent) event).getDamager() instanceof TNTPrimed) {
this.broadcast("pvp.death.explosion", defenderString + ChatColor.WHITE); this.broadcast("pvp.death.explosion", defenderString + ChatColor.WHITE);
} else if (event.getCause() == DamageCause.FIRE || event.getCause() == DamageCause.FIRE_TICK } else if (event.getCause() == DamageCause.FIRE || event.getCause() == DamageCause.FIRE_TICK
|| event.getCause() == DamageCause.LAVA || event.getCause() == DamageCause.LIGHTNING) { || event.getCause() == DamageCause.LAVA || event.getCause() == DamageCause.LIGHTNING) {
this.broadcast("pvp.death.fire", defenderString); this.broadcast("pvp.death.fire", defenderString);
} else if (event.getCause() == DamageCause.DROWNING) { } else if (event.getCause() == DamageCause.DROWNING) {
@ -1226,7 +1194,7 @@ public class Warzone {
} }
playerTeam.resetSign(); playerTeam.resetSign();
} }
private void handleTeamLoss(Team losingTeam, Player player) { private void handleTeamLoss(Team losingTeam, Player player) {
StringBuilder teamScores = new StringBuilder(); StringBuilder teamScores = new StringBuilder();
List<Team> winningTeams = new ArrayList<Team>(teams.size()); List<Team> winningTeams = new ArrayList<Team>(teams.size());
@ -1311,12 +1279,51 @@ public class Warzone {
War.war.getLogger().log(Level.INFO, "Last player left warzone {0}. Warzone blocks resetting automatically...", new Object[] {this.getName()}); War.war.getLogger().log(Level.INFO, "Last player left warzone {0}. Warzone blocks resetting automatically...", new Object[] {this.getName()});
} }
} }
this.autoTeamBalance();
WarPlayerLeaveEvent event1 = new WarPlayerLeaveEvent(player.getName()); WarPlayerLeaveEvent event1 = new WarPlayerLeaveEvent(player.getName());
War.war.getServer().getPluginManager().callEvent(event1); War.war.getServer().getPluginManager().callEvent(event1);
} }
} }
/**
* Moves players from team to team if the player size delta is greater than or equal to 2.
* Only works for autoassign zones.
*/
private void autoTeamBalance() {
if (!this.getWarzoneConfig().getBoolean(WarzoneConfig.AUTOASSIGN)) {
return;
}
boolean rerun = false;
for (Team team1 : this.teams) {
for (Team team2 : this.teams) {
if (team1 == team2) {
continue;
}
int t1p = team1.getPlayers().size();
int t2p = team2.getPlayers().size();
if (t1p - t2p >= 2) {
Player eject = team1.getPlayers().get(killSeed.nextInt(t1p));
team1.removePlayer(eject);
team1.resetSign();
this.assign(eject, team2);
rerun = true;
break;
} else if (t2p - t1p >= 2) {
Player eject = team2.getPlayers().get(killSeed.nextInt(t2p));
team2.removePlayer(eject);
team2.resetSign();
this.assign(eject, team1);
rerun = true;
break;
}
}
}
if (rerun) {
this.autoTeamBalance();
}
}
public boolean isEnemyTeamFlagBlock(Team playerTeam, Block block) { public boolean isEnemyTeamFlagBlock(Team playerTeam, Block block) {
for (Team team : this.teams) { for (Team team : this.teams) {
if (!team.getName().equals(playerTeam.getName()) && team.isTeamFlagBlock(block)) { if (!team.getName().equals(playerTeam.getName()) && team.isTeamFlagBlock(block)) {
@ -1325,7 +1332,7 @@ public class Warzone {
} }
return false; return false;
} }
public boolean isFlagBlock(Block block) { public boolean isFlagBlock(Block block) {
for (Team team : this.teams) { for (Team team : this.teams) {
if (team.isTeamFlagBlock(block)) { if (team.isTeamFlagBlock(block)) {
@ -1343,7 +1350,7 @@ public class Warzone {
} }
return null; return null;
} }
public boolean isBombBlock(Block block) { public boolean isBombBlock(Block block) {
for (Bomb bomb : this.bombs) { for (Bomb bomb : this.bombs) {
if (bomb.isBombBlock(block.getLocation())) { if (bomb.isBombBlock(block.getLocation())) {
@ -1361,7 +1368,7 @@ public class Warzone {
} }
return null; return null;
} }
public boolean isCakeBlock(Block block) { public boolean isCakeBlock(Block block) {
for (Cake cake : this.cakes) { for (Cake cake : this.cakes) {
if (cake.isCakeBlock(block.getLocation())) { if (cake.isCakeBlock(block.getLocation())) {
@ -1413,13 +1420,13 @@ public class Warzone {
public Bomb getBombForThief(Player thief) { public Bomb getBombForThief(Player thief) {
return this.bombThieves.get(thief.getUniqueId()); return this.bombThieves.get(thief.getUniqueId());
} }
// Cake
public void removeBombThief(Player thief) { public void removeBombThief(Player thief) {
this.bombThieves.remove(thief.getUniqueId()); this.bombThieves.remove(thief.getUniqueId());
} }
// Cake
public void addCakeThief(Cake cake, Player cakeThief) { public void addCakeThief(Cake cake, Player cakeThief) {
this.cakeThieves.put(cakeThief.getUniqueId(), cake); this.cakeThieves.put(cakeThief.getUniqueId(), cake);
WarPlayerThiefEvent event1 = new WarPlayerThiefEvent(cakeThief, WarPlayerThiefEvent.StolenObject.CAKE); WarPlayerThiefEvent event1 = new WarPlayerThiefEvent(cakeThief, WarPlayerThiefEvent.StolenObject.CAKE);
@ -1452,7 +1459,7 @@ public class Warzone {
} }
return false; return false;
} }
public void handleScoreCapReached(String winnersStr) { public void handleScoreCapReached(String winnersStr) {
// Score cap reached. Reset everything. // Score cap reached. Reset everything.
this.isEndOfGame = true; this.isEndOfGame = true;
@ -1462,7 +1469,7 @@ public class Warzone {
} }
WarScoreCapEvent event1 = new WarScoreCapEvent(winningTeams); WarScoreCapEvent event1 = new WarScoreCapEvent(winningTeams);
War.war.getServer().getPluginManager().callEvent(event1); War.war.getServer().getPluginManager().callEvent(event1);
for (Team t : this.getTeams()) { for (Team t : this.getTeams()) {
if (War.war.isSpoutServer()) { if (War.war.isSpoutServer()) {
for (Player p : t.getPlayers()) { for (Player p : t.getPlayers()) {
@ -1480,7 +1487,7 @@ public class Warzone {
String winnersStrAndExtra = "Score cap reached. Game is over! Winning team(s): " + winnersStr; String winnersStrAndExtra = "Score cap reached. Game is over! Winning team(s): " + winnersStr;
winnersStrAndExtra += ". Resetting warzone and your inventory..."; winnersStrAndExtra += ". Resetting warzone and your inventory...";
t.teamcast(winnersStrAndExtra); t.teamcast(winnersStrAndExtra);
double ecoReward = t.getTeamConfig().resolveDouble(TeamConfig.ECOREWARD); double ecoReward = t.getTeamConfig().resolveDouble(TeamConfig.ECOREWARD);
boolean doEcoReward = ecoReward != 0 && War.war.getEconomy() != null; boolean doEcoReward = ecoReward != 0 && War.war.getEconomy() != null;
for (Iterator<Player> it = t.getPlayers().iterator(); it.hasNext();) { for (Iterator<Player> it = t.getPlayers().iterator(); it.hasNext();) {
Player tp = it.next(); Player tp = it.next();
@ -1526,10 +1533,7 @@ public class Warzone {
} }
public boolean isDeadMan(String playerName) { public boolean isDeadMan(String playerName) {
if (this.deadMenInventories.containsKey(playerName)) { return this.deadMenInventories.containsKey(playerName);
return true;
}
return false;
} }
public void restoreDeadmanInventory(Player player) { public void restoreDeadmanInventory(Player player) {
@ -1539,14 +1543,14 @@ public class Warzone {
} }
} }
public void setRallyPoint(Location location) {
this.rallyPoint = location;
}
public Location getRallyPoint() { public Location getRallyPoint() {
return this.rallyPoint; return this.rallyPoint;
} }
public void setRallyPoint(Location location) {
this.rallyPoint = location;
}
public void unload() { public void unload() {
War.war.log("Unloading zone " + this.getName() + "...", Level.INFO); War.war.log("Unloading zone " + this.getName() + "...", Level.INFO);
for (Team team : this.getTeams()) { for (Team team : this.getTeams()) {
@ -1572,10 +1576,7 @@ public class Warzone {
teamsWithEnough++; teamsWithEnough++;
} }
} }
if (teamsWithEnough >= this.getWarzoneConfig().getInt(WarzoneConfig.MINTEAMS)) { return teamsWithEnough >= this.getWarzoneConfig().getInt(WarzoneConfig.MINTEAMS);
return true;
}
return false;
} }
/** /**
@ -1616,8 +1617,6 @@ public class Warzone {
tryCallDelayedPlayers(); tryCallDelayedPlayers();
} }
// prevent tryCallDelayedPlayers from being recursively called by Warzone#assign
private boolean activeDelayedCall = false;
private void tryCallDelayedPlayers() { private void tryCallDelayedPlayers() {
if (activeDelayedCall || (!isEnoughPlayers() && !testEnoughPlayers(null, true))) { if (activeDelayedCall || (!isEnoughPlayers() && !testEnoughPlayers(null, true))) {
return; return;
@ -1693,7 +1692,7 @@ public class Warzone {
int i = 0; int i = 0;
Iterator<String> it = sortedNames.iterator(); Iterator<String> it = sortedNames.iterator();
while (it.hasNext()) { while (it.hasNext()) {
String name = (String) it.next(); String name = it.next();
if (i == currentIndex) { if (i == currentIndex) {
if (playerTeam.getTeamConfig().resolveBoolean(TeamConfig.PLAYERLOADOUTASDEFAULT) && name.equals("default")) { if (playerTeam.getTeamConfig().resolveBoolean(TeamConfig.PLAYERLOADOUTASDEFAULT) && name.equals("default")) {
// Use player's own inventory as loadout // Use player's own inventory as loadout
@ -1789,11 +1788,6 @@ public class Warzone {
// return gameEndLock; // return gameEndLock;
// } // }
public void setName(String newName) {
this.name = newName;
this.volume.setName(newName);
}
public HubLobbyMaterials getLobbyMaterials() { public HubLobbyMaterials getLobbyMaterials() {
return this.lobbyMaterials; return this.lobbyMaterials;
} }
@ -1824,14 +1818,14 @@ public class Warzone {
return false; return false;
} }
public void setWarzoneMaterials(WarzoneMaterials warzoneMaterials) {
this.warzoneMaterials = warzoneMaterials;
}
public WarzoneMaterials getWarzoneMaterials() { public WarzoneMaterials getWarzoneMaterials() {
return warzoneMaterials; return warzoneMaterials;
} }
public void setWarzoneMaterials(WarzoneMaterials warzoneMaterials) {
this.warzoneMaterials = warzoneMaterials;
}
public Scoreboard getScoreboard() { public Scoreboard getScoreboard() {
return scoreboard; return scoreboard;
} }
@ -1839,6 +1833,7 @@ public class Warzone {
public ScoreboardType getScoreboardType() { public ScoreboardType getScoreboardType() {
return this.getWarzoneConfig().getScoreboardType(WarzoneConfig.SCOREBOARD); return this.getWarzoneConfig().getScoreboardType(WarzoneConfig.SCOREBOARD);
} }
public boolean hasKillCount(String player) { public boolean hasKillCount(String player) {
return killCount.containsKey(player); return killCount.containsKey(player);
} }
@ -1902,7 +1897,7 @@ public class Warzone {
/** /**
* Get a list of all players in the warzone. The list is immutable. If you * Get a list of all players in the warzone. The list is immutable. If you
* need to modify the player list, you must use the per-team lists * need to modify the player list, you must use the per-team lists
* *
* @return list containing all team players. * @return list containing all team players.
*/ */
public List<Player> getPlayers() { public List<Player> getPlayers() {
@ -1915,7 +1910,7 @@ public class Warzone {
/** /**
* Get the amount of players in all teams in this warzone. * Get the amount of players in all teams in this warzone.
* *
* @return total player count * @return total player count
*/ */
public int getPlayerCount() { public int getPlayerCount() {
@ -1934,12 +1929,11 @@ public class Warzone {
return zoneCap; return zoneCap;
} }
/** /**
* Get the amount of players in all teams in this warzone. Same as * Get the amount of players in all teams in this warzone. Same as
* {@link #getPlayerCount()}, except only checks teams that the specified * {@link #getPlayerCount()}, except only checks teams that the specified
* player has permission to join. * player has permission to join.
* *
* @param target * @param target
* Player to check for permissions. * Player to check for permissions.
* @return total player count in teams the player has access to. * @return total player count in teams the player has access to.
@ -1958,7 +1952,7 @@ public class Warzone {
/** /**
* Get the total capacity of all teams in this zone. This should be * Get the total capacity of all teams in this zone. This should be
* preferred over {@link TeamConfig#TEAMSIZE} as that can differ per team. * preferred over {@link TeamConfig#TEAMSIZE} as that can differ per team.
* *
* @return capacity of all teams in this zone * @return capacity of all teams in this zone
*/ */
public int getTotalCapacity() { public int getTotalCapacity() {
@ -1973,7 +1967,7 @@ public class Warzone {
* Get the total capacity of all teams in this zone. Same as * Get the total capacity of all teams in this zone. Same as
* {@link #getTotalCapacity()}, except only checks teams that the specified * {@link #getTotalCapacity()}, except only checks teams that the specified
* player has permission to join. * player has permission to join.
* *
* @param target * @param target
* Player to check for permissions. * Player to check for permissions.
* @return capacity of teams the player has access to. * @return capacity of teams the player has access to.
@ -1992,7 +1986,7 @@ public class Warzone {
/** /**
* Check if all teams are full. * Check if all teams are full.
* *
* @return true if all teams are full, false otherwise. * @return true if all teams are full, false otherwise.
*/ */
public boolean isFull() { public boolean isFull() {
@ -2002,7 +1996,7 @@ public class Warzone {
/** /**
* Check if all teams are full. Same as {@link #isFull()}, except only * Check if all teams are full. Same as {@link #isFull()}, except only
* checks teams that the specified player has permission to join. * checks teams that the specified player has permission to join.
* *
* @param target * @param target
* Player to check for permissions. * Player to check for permissions.
* @return true if all teams are full, false otherwise. * @return true if all teams are full, false otherwise.
@ -2116,12 +2110,20 @@ public class Warzone {
public boolean isThief(Player suspect) { public boolean isThief(Player suspect) {
return this.isFlagThief(suspect) || this.isBombThief(suspect) || this.isCakeThief(suspect); return this.isFlagThief(suspect) || this.isBombThief(suspect) || this.isCakeThief(suspect);
} }
public boolean getPvpReady() {
return this.pvpReady;
}
public void setPvpReady(boolean ready) { public void setPvpReady(boolean ready) {
this.pvpReady = ready; this.pvpReady = ready;
} }
public boolean getPvpReady() { public enum LeaveCause {
return this.pvpReady; COMMAND, DISCONNECT, SCORECAP, RESET;
public boolean useRallyPoint() {
return this == SCORECAP;
}
} }
} }