package net.knarcraft.dropper.arena; import net.knarcraft.dropper.Dropper; import net.knarcraft.dropper.container.SerializableUUID; import net.knarcraft.dropper.util.StringSanitizer; import org.bukkit.configuration.serialization.ConfigurationSerializable; import org.bukkit.entity.Player; import org.jetbrains.annotations.NotNull; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.UUID; import java.util.logging.Level; /** * A sorted group of arenas that must be completed in sequence */ public class DropperArenaGroup implements ConfigurationSerializable { /** * The unique id for this group of arenas */ private final UUID groupId; /** * The unique name for this group of arenas */ private final String groupName; /** * The arenas in this group, ordered from stage 1 to stage n */ private final List arenas; /** * Instantiates a new dropper arena group * * @param groupName

The name of this group

*/ public DropperArenaGroup(@NotNull String groupName) { this.groupId = UUID.randomUUID(); this.groupName = groupName; this.arenas = new ArrayList<>(); } /** * Instantiates a new dropper arena group * * @param groupId

The unique id of this group

* @param groupName

The name of this group

* @param arenas

The arenas in this group

*/ private DropperArenaGroup(@NotNull UUID groupId, @NotNull String groupName, @NotNull List arenas) { this.groupId = groupId; this.groupName = groupName; this.arenas = new ArrayList<>(arenas); } /** * Gets the id of this dropper arena group * * @return

The id of this group

*/ public @NotNull UUID getGroupId() { return this.groupId; } /** * Gets the name of this dropper arena group * * @return

The name of this group

*/ public @NotNull String getGroupName() { return this.groupName; } /** * Gets the arenas contained in this group in the correct order * * @return

The ids of the arenas in this group

*/ public @NotNull List getArenas() { return new ArrayList<>(arenas); } /** * Removes the given dropper arena from this group * * @param arenaId

The id of the dropper arena to remove

*/ public void removeArena(UUID arenaId) { this.arenas.remove(arenaId); } /** * Adds an arena to the end of this group * * @param arenaId

The arena to add to this group

*/ public void addArena(UUID arenaId) { addArena(arenaId, this.arenas.size()); } /** * Adds an arena to the end of this group * * @param arenaId

The arena to add to this group

* @param index

The index to put the arena in

*/ public void addArena(UUID arenaId, int index) { // Make sure we don't have duplicates if (!this.arenas.contains(arenaId)) { this.arenas.add(index, arenaId); } } /** * Checks whether the given player has beaten all arenas in this group on the given game-mode * * @param gameMode

The game-mode to check

* @param player

The player to check

* @return

True if the player has beaten all arenas, false otherwise

*/ public boolean hasBeatenAll(ArenaGameMode gameMode, Player player) { DropperArenaHandler arenaHandler = Dropper.getInstance().getArenaHandler(); for (UUID anArenaId : this.getArenas()) { DropperArena dropperArena = arenaHandler.getArena(anArenaId); if (dropperArena == null) { // The arena would only be null if the arena has been deleted, but not removed from this group Dropper.getInstance().getLogger().log(Level.WARNING, "The dropper group " + this.getGroupName() + " contains the arena id " + anArenaId + " which is not a valid arena id!"); continue; } if (dropperArena.getData().hasNotCompleted(gameMode, player)) { return false; } } return true; } /** * Gets whether the given player can play the given arena part of this group, on the given game-mode * * @param gameMode

The game-mode the player is trying to play

* @param player

The player to check

* @param arenaId

The id of the arena in this group to check

* @return

True if the player is allowed to play the arena

* @throws IllegalArgumentException

If checking an arena not in this group

*/ public boolean canPlay(ArenaGameMode gameMode, Player player, UUID arenaId) throws IllegalArgumentException { if (!this.arenas.contains(arenaId)) { throw new IllegalArgumentException("Cannot check for playability for arena not in this group!"); } DropperArenaHandler arenaHandler = Dropper.getInstance().getArenaHandler(); for (UUID anArenaId : this.getArenas()) { // If the target arena is reached, allow, as all previous arenas must have been cleared if (arenaId.equals(anArenaId)) { return true; } DropperArena dropperArena = arenaHandler.getArena(anArenaId); if (dropperArena == null) { // The arena would only be null if the arena has been deleted, but not removed from this group Dropper.getInstance().getLogger().log(Level.WARNING, String.format("The dropper group %s contains the" + " arena id %s which is not a valid arena id!", this.getGroupName(), anArenaId)); continue; } // This is a lower-numbered arena the player has yet to complete if (dropperArena.getData().hasNotCompleted(gameMode, player)) { return false; } } return false; } /** * Gets this group's name, but sanitized * * @return

The sanitized group name

*/ public @NotNull String getGroupNameSanitized() { return StringSanitizer.sanitizeArenaName(this.getGroupName()); } /** * Swaps the arenas at the given indices * * @param index1

The index of the first arena to swap

* @param index2

The index of the second arena to swap

*/ public void swapArenas(int index1, int index2) { // Change nothing if not a valid request if (index1 == index2 || index1 < 0 || index2 < 0 || index1 >= this.arenas.size() || index2 >= this.arenas.size()) { return; } // Swap the two arena ids UUID temporaryValue = this.arenas.get(index2); this.arenas.set(index2, this.arenas.get(index1)); this.arenas.set(index1, temporaryValue); } @NotNull @Override public Map serialize() { Map data = new HashMap<>(); data.put("groupId", new SerializableUUID(this.groupId)); data.put("groupName", this.groupName); List serializableArenas = new ArrayList<>(); for (UUID arenaId : arenas) { serializableArenas.add(new SerializableUUID(arenaId)); } data.put("arenas", serializableArenas); return data; } /** * Deserializes the given data * * @param data

The data to deserialize

* @return

The deserialized arena group

*/ @SuppressWarnings({"unused", "unchecked"}) public static @NotNull DropperArenaGroup deserialize(@NotNull Map data) { UUID id = ((SerializableUUID) data.get("groupId")).uuid(); String name = (String) data.get("groupName"); List serializableArenas = (List) data.get("arenas"); List arenas = new ArrayList<>(); for (SerializableUUID arenaId : serializableArenas) { arenas.add(arenaId.uuid()); } return new DropperArenaGroup(id, name, arenas); } @Override public boolean equals(Object other) { if (!(other instanceof DropperArenaGroup otherGroup)) { return false; } return this.getGroupNameSanitized().equals(otherGroup.getGroupNameSanitized()); } }