package net.knarcraft.minigames.arena; import net.knarcraft.minigames.MiniGames; import net.knarcraft.minigames.util.StringSanitizer; import org.jetbrains.annotations.NotNull; import org.jetbrains.annotations.Nullable; import java.util.HashMap; import java.util.HashSet; import java.util.Map; import java.util.Set; import java.util.UUID; import java.util.logging.Level; /** * An interface describing a generic arena handler * * @param

The type of arena stored

* @param

The type of arena group stored

*/ public abstract class ArenaHandler> { protected Map arenas = new HashMap<>(); protected Map arenaGroups = new HashMap<>(); protected Map arenaNameLookup = new HashMap<>(); private final ArenaPlayerRegistry playerRegistry; /** * Instantiates a new arena handler * * @param playerRegistry

The registry keeping track of player sessions

*/ public ArenaHandler(ArenaPlayerRegistry playerRegistry) { this.playerRegistry = playerRegistry; } /** * Gets all arenas that are within a group * * @return

All arenas in a group

*/ public @NotNull Set getArenasInAGroup() { Set arenas = new HashSet<>(); for (UUID arenaId : arenaGroups.keySet()) { arenas.add(this.arenas.get(arenaId)); } return arenas; } /** * Gets a copy of all arena groups * * @return

All arena groups

*/ public Set getAllGroups() { return new HashSet<>(arenaGroups.values()); } /** * Gets the group the given arena belongs to * * @param arenaId

The id of the arena to get the group of

* @return

The group the arena belongs to, or null if not in a group

*/ public @Nullable S getGroup(@NotNull UUID arenaId) { return this.arenaGroups.get(arenaId); } /** * Sets the group for the given arena * * @param arenaId

The id of the arena to change

* @param arenaGroup

The group to add the arena to, or null to remove the current group

*/ public void setGroup(@NotNull UUID arenaId, @Nullable S arenaGroup) { if (arenaGroup == null) { // No need to remove something non-existing if (!this.arenaGroups.containsKey(arenaId)) { return; } // Remove the existing group S oldGroup = this.arenaGroups.remove(arenaId); oldGroup.removeArena(arenaId); } else { // Make sure to remove the arena from the old group's internal tracking if (this.arenaGroups.containsKey(arenaId)) { this.arenaGroups.remove(arenaId).removeArena(arenaId); } this.arenaGroups.put(arenaId, arenaGroup); arenaGroup.addArena(arenaId); } saveGroups(); } /** * Gets the arena group with the given name * * @param groupName

The name of the group to get

* @return

The group, or null if not found

*/ public @Nullable S getGroup(String groupName) { String sanitized = StringSanitizer.sanitizeArenaName(groupName); for (S arenaGroup : this.arenaGroups.values()) { if (arenaGroup.getGroupNameSanitized().equals(sanitized)) { return arenaGroup; } } return null; } /** * Replaces an arena's lookup name * * @param oldName

The arena's old sanitized lookup name

* @param newName

The arena's new sanitized lookup name

*/ public void updateLookupName(@NotNull String oldName, @NotNull String newName) { UUID arenaId = this.arenaNameLookup.remove(oldName); if (arenaId != null) { this.arenaNameLookup.put(newName, arenaId); } } /** * Adds a new arena * * @param arena

The arena to add

*/ public void addArena(@NotNull K arena) { this.arenas.put(arena.getArenaId(), arena); this.arenaNameLookup.put(arena.getArenaNameSanitized(), arena.getArenaId()); this.saveArenas(); } /** * Gets the arena with the given id * * @param arenaId

The id of the arena to get

* @return

The arena, or null if no arena could be found

*/ public @Nullable K getArena(@NotNull UUID arenaId) { return this.arenas.get(arenaId); } /** * Gets the arena with the given name * * @param arenaName

The arena to get

* @return

The arena with the given name, or null if not found

*/ public @Nullable K getArena(@NotNull String arenaName) { return this.arenas.get(this.arenaNameLookup.get(StringSanitizer.sanitizeArenaName(arenaName))); } /** * Gets all known arenas * * @return

All known arenas

*/ public @NotNull Map getArenas() { return new HashMap<>(this.arenas); } /** * Removes the given arena * * @param arena

The arena to remove

*/ public void removeArena(@NotNull K arena) { UUID arenaId = arena.getArenaId(); this.playerRegistry.removeForArena(arena, false); this.arenas.remove(arenaId); this.arenaNameLookup.remove(arena.getArenaNameSanitized()); this.arenaGroups.remove(arenaId); if (!arena.removeData()) { MiniGames.log(Level.WARNING, "Unable to remove arena data file " + arenaId + ".yml. " + "You must remove it manually!"); } this.saveArenas(); } /** * Stores the data for the given arena * * @param arenaId

The id of the arena whose data should be saved

*/ public void saveData(UUID arenaId) { K arena = getArena(arenaId); if (arena != null) { if (!arena.saveData()) { MiniGames.log(Level.WARNING, "Unable to save data for arena with id " + arenaId + " because of a write exception!"); } } else { MiniGames.log(Level.WARNING, "Unable to save data for arena with id " + arenaId + " because the arena could not be found!"); } } /** * Saves all current groups to disk */ public abstract void saveGroups(); /** * Loads all groups from disk */ protected abstract void loadGroups(); /** * Loads all arenas and groups from disk */ public void load() { loadArenas(); loadGroups(); } /** * Saves all current arenas to disk */ public abstract void saveArenas(); /** * Loads all arenas from disk */ protected abstract void loadArenas(); }