package cryodex.modules.starwarslcg; import java.net.URL; import java.util.ArrayList; import java.util.Collections; import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.Set; import java.util.TreeSet; import javax.swing.Icon; import javax.swing.ImageIcon; import javax.swing.JOptionPane; import cryodex.CryodexController; import cryodex.CryodexController.Modules; import cryodex.Main; import cryodex.Player; import cryodex.modules.Module; import cryodex.modules.Tournament; import cryodex.xml.XMLObject; import cryodex.xml.XMLUtils; import cryodex.xml.XMLUtils.Element; public class SWLCGTournament implements XMLObject, Tournament { public enum InitialSeedingEnum { RANDOM, BY_GROUP, IN_ORDER; } private final List<SWLCGRound> rounds; private List<SWLCGPlayer> players; private final InitialSeedingEnum seedingEnum; private final SWLCGTournamentGUI tournamentGUI; private String name; private boolean startAsSingleElimination = false; public SWLCGTournament(Element tournamentElement) { this.players = new ArrayList<>(); this.rounds = new ArrayList<>(); seedingEnum = InitialSeedingEnum.RANDOM; tournamentGUI = new SWLCGTournamentGUI(this); String playerIDs = tournamentElement.getStringFromChild("PLAYERS"); Module m = Modules.getModuleByName(getModuleName()); for (String s : playerIDs.split(",")) { Player p = CryodexController.getPlayerByID(s); if (p != null) { SWLCGPlayer xp = (SWLCGPlayer) p.getModuleInfoByModule(m); if (xp != null) { players.add(xp); } } } Element roundElement = tournamentElement.getChild("ROUNDS"); for (Element e : roundElement.getChildren()) { rounds.add(new SWLCGRound(e, this)); } name = tournamentElement.getStringFromChild("NAME"); int counter = 1; for (SWLCGRound r : rounds) { if (r.isElimination()) { getTournamentGUI().getRoundTabbedPane() .addSingleEliminationTab(r.getMatches().size() * 2, r.getPanel()); } else { getTournamentGUI().getRoundTabbedPane().addSwissTab(counter, r.getPanel()); counter++; } } getTournamentGUI().getRankingTable().setPlayers(getAllSWLCGPlayers()); } public SWLCGTournament(String name, List<SWLCGPlayer> players, InitialSeedingEnum seedingEnum, boolean isSingleElimination) { this.name = name; this.players = new ArrayList<>(players); this.rounds = new ArrayList<>(); this.seedingEnum = seedingEnum; this.startAsSingleElimination = isSingleElimination; tournamentGUI = new SWLCGTournamentGUI(this); } public SWLCGRound getLatestRound() { if (rounds == null || rounds.isEmpty()) { return null; } else { return rounds.get(rounds.size() - 1); } } public int getRoundNumber(SWLCGRound round) { int count = 0; for (SWLCGRound r : rounds) { count++; if (r == round) { return count; } } return 0; } public SWLCGRound getRound(int i) { if (rounds == null) { return null; } else { return rounds.get(i); } } public SWLCGRound getSelectedRound() { if (rounds == null) { return null; } else { return getAllRounds().get( getTournamentGUI().getRoundTabbedPane().getSelectedIndex()); } } public List<SWLCGRound> getAllRounds() { return rounds; } @Override public int getRoundCount() { if (rounds == null) { return 0; } else { return rounds.size(); } } @Override public void setPlayers(List<Player> players) { List<SWLCGPlayer> xwPlayers = new ArrayList<>(); for (Player p : players) { SWLCGPlayer xp = new SWLCGPlayer(p); xwPlayers.add(xp); } setSWLCGPlayer(xwPlayers); } @Override public List<Player> getPlayers() { List<Player> players = new ArrayList<Player>(); for (SWLCGPlayer xp : getSWLCGPlayers()) { players.add(xp.getPlayer()); } return players; } public List<SWLCGPlayer> getSWLCGPlayers() { return players; } public void setSWLCGPlayer(List<SWLCGPlayer> players) { this.players = players; } public Set<SWLCGPlayer> getAllSWLCGPlayers() { Set<SWLCGPlayer> allPlayers = new TreeSet<SWLCGPlayer>(new SWLCGComparator(this, SWLCGComparator.rankingCompare)); for (SWLCGRound r : getAllRounds()) { for (SWLCGMatch m : r.getMatches()) { if (m.isBye()) { allPlayers.add(m.getPlayer1()); } else { allPlayers.add(m.getPlayer1()); if (m.getPlayer2() != null) { allPlayers.add(m.getPlayer2()); } } } } allPlayers.addAll(players); return allPlayers; } @Override public Set<Player> getAllPlayers() { Set<Player> players = new TreeSet<Player>(); for (SWLCGPlayer xp : getAllSWLCGPlayers()) { players.add(xp.getPlayer()); } return players; } @Override public SWLCGTournamentGUI getTournamentGUI() { return tournamentGUI; } @Override public String getName() { return name; } @Override public void setName(String name) { this.name = name; } @Override public void updateVisualOptions() { if (CryodexController.isLoading == false) { for (SWLCGRound r : getAllRounds()) { r.getPanel().resetGamePanels(true); } } } @Override public boolean generateNextRound() { if (getLatestRound().isComplete() == false) { JOptionPane .showMessageDialog(Main.getInstance(), "Current round is not complete. Please complete all matches before continuing"); return false; } if (getLatestRound().isValid() == false) { JOptionPane .showMessageDialog( Main.getInstance(), "At least one tournamnt result is not correct. Check if points are backwards or a result should be a modified win or tie."); return false; } if (getLatestRound().isElimination()) { if (getLatestRound().getMatches().size() == 1) { JOptionPane .showMessageDialog(Main.getInstance(), "Final tournament complete. No more rounds will be generated."); return false; } generateSingleEliminationMatches(getLatestRound().getMatches() .size()); } else { generateRound(getAllRounds().size() + 1); } return true; } @Override public void cancelRound(int roundNumber) { if (rounds.size() >= roundNumber) { // If we are generating a past round. Clear all existing rounds that // will be erased. while (rounds.size() >= roundNumber) { int index = rounds.size() - 1; SWLCGRound roundToRemove = rounds.get(index); for (SWLCGMatch m : roundToRemove.getMatches()) { m.setBye(false); m.setPlayer1(null); m.setPlayer2(null); m.setGame1Result(null); m.setGame2Result(null); } rounds.remove(roundToRemove); getTournamentGUI().getRoundTabbedPane().remove(index); } } } @Override public void generateRound(int roundNumber) { // if trying to skip a round...stop it if (roundNumber > rounds.size() + 1) { throw new IllegalArgumentException(); } cancelRound(roundNumber); List<SWLCGMatch> matches; if (roundNumber == 1) { matches = new ArrayList<SWLCGMatch>(); List<SWLCGPlayer> tempList = new ArrayList<>(); tempList.addAll(getSWLCGPlayers()); List<SWLCGPlayer> firstRoundByePlayers = new ArrayList<>(); for (SWLCGPlayer p : tempList) { if (p.isFirstRoundBye()) { firstRoundByePlayers.add(p); } } tempList.removeAll(firstRoundByePlayers); if (seedingEnum == InitialSeedingEnum.IN_ORDER) { while (tempList.isEmpty() == false) { SWLCGPlayer player1 = tempList.get(0); SWLCGPlayer player2 = null; tempList.remove(0); if (tempList.isEmpty() == false) { player2 = tempList.get(0); tempList.remove(0); } SWLCGMatch match = new SWLCGMatch(player1, player2); matches.add(match); } } else if (seedingEnum == InitialSeedingEnum.RANDOM) { Collections.shuffle(tempList); while (tempList.isEmpty() == false) { SWLCGPlayer player1 = tempList.get(0); SWLCGPlayer player2 = tempList.get(tempList.size() - 1); tempList.remove(player1); if (player1 == player2) { player2 = null; } else { tempList.remove(player2); } SWLCGMatch match = new SWLCGMatch(player1, player2); matches.add(match); } } else if (seedingEnum == InitialSeedingEnum.BY_GROUP) { Map<String, List<SWLCGPlayer>> playerMap = new HashMap<String, List<SWLCGPlayer>>(); // Add players to map for (SWLCGPlayer p : tempList) { List<SWLCGPlayer> playerList = playerMap.get(p.getPlayer() .getGroupName()); if (playerList == null) { playerList = new ArrayList<>(); String groupName = p.getPlayer().getGroupName() == null ? "" : p.getPlayer().getGroupName(); playerMap.put(groupName, playerList); } playerList.add(p); } // Shuffle up the lists List<String> seedValues = new ArrayList<>(playerMap.keySet()); Collections.shuffle(seedValues); // Shuffle each group list for (List<SWLCGPlayer> list : playerMap.values()) { Collections.shuffle(list); } // ///////////// // Add new algorythm here // ///////////// SWLCGPlayer p1 = null; SWLCGPlayer p2 = null; while (seedValues.isEmpty() == false) { int i = 0; while (i < seedValues.size()) { if (p1 == null) { p1 = playerMap.get(seedValues.get(i)).get(0); } else { p2 = playerMap.get(seedValues.get(i)).get(0); matches.add(new SWLCGMatch(p1, p2)); p1 = null; p2 = null; } playerMap.get(seedValues.get(i)).remove(0); if (playerMap.get(seedValues.get(i)).isEmpty()) { seedValues.remove(i); } else { i++; } } Collections.shuffle(seedValues); } if (p1 != null) { matches.add(new SWLCGMatch(p1, null)); } } for (SWLCGPlayer p : firstRoundByePlayers) { matches.add(new SWLCGMatch(p, null)); } } else { matches = getMatches(getSWLCGPlayers()); } SWLCGRound r = new SWLCGRound(matches, this, roundNumber); rounds.add(r); if (roundNumber == 1 && startAsSingleElimination && (matches.size() == 1 || matches.size() == 2 || matches.size() == 4 || matches.size() == 8 || matches.size() == 16 || matches.size() == 32)) { r.setElimination(true); getTournamentGUI().getRoundTabbedPane().addSingleEliminationTab( r.getMatches().size() * 2, r.getPanel()); } else { getTournamentGUI().getRoundTabbedPane().addSwissTab(roundNumber, r.getPanel()); } getTournamentGUI().getRankingTable().setPlayers(getAllSWLCGPlayers()); } private List<SWLCGMatch> getMatches(List<SWLCGPlayer> userList) { List<SWLCGMatch> matches = new ArrayList<SWLCGMatch>(); List<SWLCGPlayer> tempList = new ArrayList<SWLCGPlayer>(); tempList.addAll(userList); Collections.sort(tempList, new SWLCGComparator(this, SWLCGComparator.pairingCompare)); SWLCGMatch byeMatch = null; // Setup the bye match if necessary // The player to get the bye is the lowest ranked player who has not had // a bye yet or who has the fewest byes if (tempList.size() % 2 == 1) { SWLCGPlayer byeUser = null; int byUserCounter = 1; int minByes = 0; try { while (byeUser == null || byeUser.getByes(this) > minByes || (byeUser.getMatches(this) != null && byeUser .getMatches(this) .get(byeUser.getMatches(this).size() - 1) .isBye())) { if (byUserCounter > tempList.size()) { minByes++; byUserCounter = 1; } byeUser = tempList.get(tempList.size() - byUserCounter); byUserCounter++; } } catch (ArrayIndexOutOfBoundsException e) { byeUser = tempList.get(tempList.size() - 1); } byeMatch = new SWLCGMatch(byeUser, null); tempList.remove(byeUser); } matches = new SWLCGRandomMatchGeneration(this, tempList).generateMatches(); if (SWLCGMatch.hasDuplicate(matches)) { JOptionPane .showMessageDialog(Main.getInstance(), "Unable to resolve duplicate matches. Please review for best course of action."); } // Add the bye match at the end if (byeMatch != null) { matches.add(byeMatch); } return matches; } @Override public void generateSingleEliminationMatches(int cutSize) { List<SWLCGMatch> matches = new ArrayList<SWLCGMatch>(); if (getLatestRound().isElimination()) { matches = SWLCGEliminationGeneration.generateNextEliminationRound(this); } else { matches =SWLCGEliminationGeneration.setupInitialBracket(this, cutSize); } SWLCGRound r = new SWLCGRound(matches, this, null); r.setElimination(true); rounds.add(r); getTournamentGUI().getRoundTabbedPane().addSingleEliminationTab( cutSize, r.getPanel()); CryodexController.saveData(); } @Override public StringBuilder appendXML(StringBuilder sb) { String playerString = ""; String seperator = ""; for (SWLCGPlayer p : players) { playerString += seperator + p.getPlayer().getSaveId(); seperator = ","; } XMLUtils.appendObject(sb, "PLAYERS", playerString); XMLUtils.appendList(sb, "ROUNDS", "ROUND", getAllRounds()); XMLUtils.appendObject(sb, "NAME", name); XMLUtils.appendObject(sb, "MODULE", Modules.SWLCG.getName()); return sb; } @Override public void startTournament() { generateRound(1); } @Override public void addPlayer(Player p) { for(SWLCGRound r : getAllRounds()){ for(SWLCGMatch m : r.getMatches()){ if(m.getPlayer1().getPlayer().equals(p)){ getSWLCGPlayers().add(m.getPlayer1()); return; } else if(m.getPlayer2() != null && m.getPlayer2().getPlayer().equals(p)) { getSWLCGPlayers().add(m.getPlayer2()); return; } } } SWLCGPlayer xPlayer = new SWLCGPlayer(p); getSWLCGPlayers().add(xPlayer); } @Override public void dropPlayer(Player p) { SWLCGPlayer xPlayer = null; for (SWLCGPlayer xp : getSWLCGPlayers()) { if (xp.getPlayer() == p) { xPlayer = xp; break; } } if (xPlayer != null) { getSWLCGPlayers().remove(xPlayer); } resetRankingTable(); } @Override public void resetRankingTable() { getTournamentGUI().getRankingTable().setPlayers(getAllSWLCGPlayers()); } @Override public Icon getIcon() { URL imgURL = SWLCGTournament.class.getResource("l.png"); if (imgURL == null) { System.out.println("fail!!!!!!!!!!"); } ImageIcon icon = new ImageIcon(imgURL); return icon; } @Override public String getModuleName() { return Modules.SWLCG.getName(); } @Override public void massDropPlayers(int minScore, int maxCount) { // TODO Auto-generated method stub } @Override public void massDropPlayers(List<Player> playersToDrop) { // TODO Auto-generated method stub } }