package net.knarcraft.minigames.util; import net.knarcraft.knarlib.formatting.StringFormatter; import net.knarcraft.knarlib.util.MaterialHelper; import net.knarcraft.minigames.MiniGames; import net.knarcraft.minigames.arena.reward.CommandReward; import net.knarcraft.minigames.arena.reward.EconomyReward; import net.knarcraft.minigames.arena.reward.ItemReward; import net.knarcraft.minigames.arena.reward.PermissionReward; import net.knarcraft.minigames.arena.reward.Reward; import net.knarcraft.minigames.arena.reward.RewardType; import net.knarcraft.minigames.config.MiniGameMessage; import org.bukkit.Bukkit; import org.bukkit.Material; import org.bukkit.World; import org.bukkit.entity.Player; import org.bukkit.inventory.ItemStack; import org.jetbrains.annotations.NotNull; import org.jetbrains.annotations.Nullable; import java.util.Collection; import java.util.UUID; /** * A helper class for getting the reward specified in user input */ public final class RewardHelper { private RewardHelper() { } /** * Grants the given rewards to the given player * * @param player

The player to reward

* @param rewards

The rewards to give

*/ public static void grantRewards(@NotNull Player player, @NotNull Collection rewards) { StringFormatter stringFormatter = MiniGames.getInstance().getStringFormatter(); for (Reward reward : rewards) { boolean granted = reward.grant(player); if (granted) { stringFormatter.displaySuccessMessage(player, reward.getGrantMessage()); } } } /** * Parses input describing a reward * * @param player

The player that specified the reward

* @param typeString

The string given to specify the reward type

* @param firstArgument

The first reward argument given by the player, or null

* @param secondArgument

The second reward argument given by the player, or null

* @param allArguments

A list of all the reward arguments, in case the reward is a command reward

* @return

The parsed reward, or null if some input was invalid

*/ public static @Nullable Reward parseRewardInput(@NotNull Player player, @NotNull String typeString, @Nullable String firstArgument, @Nullable String secondArgument, @NotNull String[] allArguments) { RewardType rewardType = RewardType.getFromString(typeString); if (rewardType == null) { MiniGames.getInstance().getStringFormatter().displayErrorMessage(player, MiniGameMessage.ERROR_REWARD_TYPE_INVALID); return null; } if (rewardType != RewardType.ITEM && firstArgument == null) { return null; } try { return switch (rewardType) { case COMMAND -> new CommandReward(getArrayAsString(allArguments)); case ECONOMY -> new EconomyReward(getDouble(firstArgument)); case PERMISSION -> new PermissionReward(getWorld(secondArgument), firstArgument); case ITEM -> new ItemReward(getItem(player, firstArgument, secondArgument)); }; } catch (IllegalArgumentException exception) { MiniGames.getInstance().getStringFormatter().displayErrorMessage(player, exception.getMessage()); return null; } } /** * Gets a double from the given input * * @param input

The input representing a double

* @return

The double specified

* @throws IllegalArgumentException

If the input is not a number or is not positive

*/ private static double getDouble(@NotNull String input) throws IllegalArgumentException { IllegalArgumentException invalidException = new IllegalArgumentException( MiniGames.getInstance().getTranslator().getTranslatedMessage(MiniGameMessage.ERROR_INVALID_NUMBER)); try { double number = Double.parseDouble(input); if (number <= 0) { throw invalidException; } return number; } catch (NumberFormatException exception) { throw invalidException; } } /** * Gets the world specified in the given identifier * * @param worldIdentifier

A world UUID or name

* @return

The world, or null if no such world exists

*/ private static @Nullable World getWorld(@Nullable String worldIdentifier) { if (worldIdentifier == null || worldIdentifier.isBlank()) { return null; } World world; try { UUID worldId = UUID.fromString(worldIdentifier); world = Bukkit.getWorld(worldId); } catch (IllegalArgumentException exception) { world = Bukkit.getWorld(worldIdentifier); } if (world != null) { return world; } else { throw new IllegalArgumentException(MiniGames.getInstance().getTranslator().getTranslatedMessage( MiniGameMessage.ERROR_INVALID_WORLD)); } } /** * Gets an item stack according to the given input * * @param player

The player that caused this method to execute

* @param argument1

The first argument given by the player, or null

* @param argument2

The second argument given by the player, or null

* @return

An item stack as described, or the player's held item

* @throws IllegalArgumentException

If an invalid material was specified

*/ private static @NotNull ItemStack getItem(@NotNull Player player, @Nullable String argument1, @Nullable String argument2) throws IllegalArgumentException { if (argument1 != null) { Material material = MaterialHelper.loadMaterialString(argument1, MiniGames.getInstance().getLogger()); int amount; try { if (argument2 != null) { amount = Integer.parseInt(argument2); } else { amount = 1; } } catch (NumberFormatException exception) { amount = 1; } if (material == null || material.isAir()) { throw new IllegalArgumentException(MiniGames.getInstance().getTranslator().getTranslatedMessage( MiniGameMessage.ERROR_INVALID_MATERIAL)); } return new ItemStack(material, amount); } else { ItemStack inHand = player.getInventory().getItemInMainHand(); if (!inHand.getType().isAir()) { return inHand; } else { throw new IllegalArgumentException(MiniGames.getInstance().getTranslator().getTranslatedMessage( MiniGameMessage.ERROR_INVALID_MATERIAL)); } } } /** * Gets the string array as a space separated string * * @param array

The array to get as a string

* @return

The array as a string

*/ private static String getArrayAsString(@NotNull String[] array) { String output = String.join(" ", array); if (output.isBlank()) { throw new IllegalArgumentException(MiniGames.getInstance().getTranslator().getTranslatedMessage( MiniGameMessage.ERROR_INVALID_COMMAND_STRING)); } else { return output; } } }