diff --git a/src/main/java/top/penowl/quidproquo/Altar.java b/src/main/java/top/penowl/quidproquo/Altar.java index 1820685..13f4dec 100644 --- a/src/main/java/top/penowl/quidproquo/Altar.java +++ b/src/main/java/top/penowl/quidproquo/Altar.java @@ -5,6 +5,8 @@ import org.bukkit.Location; import org.bukkit.Material; public class Altar { + + // this thing just makes sure the multiblock is built right, doesn't check for air right now public static Boolean CheckAltar(Location location) { Boolean gold_block = location.getBlock().getType() == Material.GOLD_BLOCK; int iron_blocks = 0; diff --git a/src/main/java/top/penowl/quidproquo/Events.java b/src/main/java/top/penowl/quidproquo/Events.java index e0a1f3a..6175439 100644 --- a/src/main/java/top/penowl/quidproquo/Events.java +++ b/src/main/java/top/penowl/quidproquo/Events.java @@ -1,10 +1,13 @@ package top.penowl.quidproquo; import java.util.ArrayList; +import java.util.Arrays; import java.util.Collection; +import java.util.Collections; import java.util.List; import java.util.Map; import java.util.Random; +import java.util.UUID; import java.util.stream.Collectors; import org.bukkit.Bukkit; @@ -30,51 +33,84 @@ public final class Events implements Listener { @EventHandler public static void onPlayerInteract(PlayerInteractEvent event) { + // only right clicks please if (event.getAction() != Action.RIGHT_CLICK_BLOCK) return; + // general data that we want Block block = event.getClickedBlock(); Player player = event.getPlayer(); Location location = block.getLocation(); + // skip if we didn't click an altar if (!Altar.CheckAltar(block.getLocation())) return; + // cycle targets if player is sneaking, otherwise try to enact a ritual if (player.isSneaking()) { - // change target + // get the player's uuid + UUID playerUuid = player.getUniqueId(); + + // get a list of everyone else's uuid + List players = new ArrayList(Bukkit.getOnlinePlayers()); + List uuids = new ArrayList(); + for (Player target : players) { + if (target.getUniqueId() != playerUuid) { + uuids.add(target.getUniqueId()); + } + } + Collections.sort(uuids); + + // if the player already has a target, pick the next one, otherwise pick the first + if (QuidProQuo.instance.targets.containsKey(playerUuid)) { + UUID current_target = QuidProQuo.instance.targets.get(playerUuid); + if (Bukkit.getPlayer(QuidProQuo.instance.targets.get(playerUuid)) == null) { + QuidProQuo.instance.targets.put(playerUuid, uuids.get(0)); + } + QuidProQuo.instance.targets.put(playerUuid, uuids.get((uuids.indexOf(current_target) + 1) % uuids.size())); + } else { + QuidProQuo.instance.targets.put(playerUuid, uuids.get(0)); + } + + // friendly message + player.sendMessage(ChatColor.YELLOW + "Switched target to " + Bukkit.getPlayer(QuidProQuo.instance.targets.get(playerUuid))); } else { + // java control flow go brrrr Boolean success = false; + // we find all of the items and mobs that matter Collection items = block.getWorld().getEntitiesByClass(Item.class); ArrayList near_items = new ArrayList(); - + Collection sacrifices = block.getWorld().getEntitiesByClass(LivingEntity.class); + ArrayList near_sacrifices = new ArrayList(); for (Item item : items) { if (item.getLocation().distance(location) < 2) { near_items.add(item); } } - - Bukkit.getLogger().info(near_items.toString()); - - Collection sacrifices = block.getWorld().getEntitiesByClass(LivingEntity.class); - ArrayList near_sacrifices = new ArrayList(); - for (LivingEntity sacrifice : sacrifices) { if (sacrifice.getLocation().distance(location) < 2) { near_sacrifices.add(sacrifice); } } + // we check each ritual in order for (Ritual ritual : QuidProQuo.instance.rituals) { + + // this is where all the ingredients get collected in ArrayList> possibleItems = new ArrayList>(); ArrayList> possibleSacrifices = new ArrayList>(); Boolean failed = false; ArrayList byproducts = new ArrayList(); + + // iterate over each different part of the recipe for (Map.Entry entry : ritual.ingredients.entrySet()) { + Material material = entry.getKey(); int count = entry.getValue(); + // this just sorts out matching items List matches = near_items.stream().filter(item -> item.getItemStack().getType() == material).collect(Collectors.toList()); int matchCount = 0; for (Item item : matches) { @@ -88,6 +124,8 @@ public final class Events implements Listener { break; } } + + // same thing but mob sacrifices for (Map.Entry entry : ritual.sacrifices.entrySet()) { EntityType entityType = entry.getKey(); int count = entry.getValue(); @@ -101,42 +139,74 @@ public final class Events implements Listener { } } + // next recipe if we missed an ingredient if (failed) continue; + // delete all used items for (List itemArray : possibleItems) { for (Item item : itemArray) { item.remove(); } } + // kill all sacrifices for (List sacrificeArray : possibleSacrifices) { for (LivingEntity sacrifice : sacrificeArray) { sacrifice.setHealth(0); } } + // administer health penalty player.setHealth(player.getHealth() - ritual.health); + // shiny message player.sendMessage(ChatColor.GREEN + "You enacted a " + ritual.name + " ritual."); + // add in extra items from recipe for (ItemStack additionalByproduct : ritual.byproducts) { byproducts.add(additionalByproduct); } + // spit out extra items that were consumed for (ItemStack byproduct : byproducts) { block.getLocation().getWorld().dropItem(location.clone().add(0, 2, 0), byproduct); } + // uuid stuff to make sure we target the right person + UUID playerUuid = player.getUniqueId(); + Player otherPlayer; + List players = new ArrayList(Bukkit.getOnlinePlayers()); + List uuids = new ArrayList(); + for (Player target : players) { + if (target.getUniqueId() != playerUuid) { + uuids.add(target.getUniqueId()); + } + } + Collections.sort(uuids); + + if (!QuidProQuo.instance.targets.containsKey(playerUuid)) { + QuidProQuo.instance.targets.put(playerUuid, uuids.get(0)); + } + if (Bukkit.getPlayer(QuidProQuo.instance.targets.get(playerUuid)) == null) { + QuidProQuo.instance.targets.put(playerUuid, uuids.get(0)); + } + otherPlayer = Bukkit.getPlayer(QuidProQuo.instance.targets.get(playerUuid)); + + // backfire check, if succeeds then the ritual gets reversed if(new Random().nextDouble() >= ritual.backfire) { - ritual.execute(player, null, block.getLocation()); + ritual.execute(player, otherPlayer, block.getLocation()); } else { - ritual.execute(null, player, block.getLocation()); + ritual.execute(otherPlayer, player, block.getLocation()); } + // cool effects player.playSound(player.getLocation(), Sound.EXPLODE, 50, 0); player.getWorld().playEffect(location.clone().add(0, 1, 0), Effect.EXPLOSION_LARGE, 0); - player.getWorld().strikeLightningEffect(location.clone().add(0, 1, 0)); + if (ritual.lightning) { + player.getWorld().strikeLightningEffect(location.clone().add(0, 1, 0)); + } + // poggers success = true; break; diff --git a/src/main/java/top/penowl/quidproquo/QuidProQuo.java b/src/main/java/top/penowl/quidproquo/QuidProQuo.java index 27fc251..9ac15de 100644 --- a/src/main/java/top/penowl/quidproquo/QuidProQuo.java +++ b/src/main/java/top/penowl/quidproquo/QuidProQuo.java @@ -11,31 +11,47 @@ import top.penowl.quidproquo.rituals.HealRitual; import top.penowl.quidproquo.rituals.WoolingRitual; public class QuidProQuo extends JavaPlugin { - + + // DONT COMMENT CODE AT 3 AM + + // haha singleton go brrr public static QuidProQuo instance; + + // global list of registered rituals public ArrayList rituals = new ArrayList(); + + // dictionary of who has who selected as a ritual target public HashMap targets = new HashMap(); + // this gets run when the plugin loads @Override public void onEnable() { + + // here we register events and the singleton instance = this; - getLogger().info("Hello, SpigotMC!"); + getLogger().info("Loading rituals..."); getServer().getPluginManager().registerEvents(new Events(), this); + // register all rituals rituals.add(new HealRitual()); rituals.add(new WoolingRitual()); rituals.add(new FeedingRitual()); + // run ritual setup scripts for (Ritual ritual : rituals) { getLogger().info("Loading " + ritual.getClass().toString() + "..."); ritual.setup(); getLogger().info("Loaded a " + ritual.name + " ritual."); } + } @Override public void onDisable() { - getLogger().info("See you again, SpigotMC!"); + + // basically just handler cleanup + getLogger().info("Unloading handler..."); HandlerList.unregisterAll(this); + } } \ No newline at end of file diff --git a/src/main/java/top/penowl/quidproquo/Ritual.java b/src/main/java/top/penowl/quidproquo/Ritual.java index c19a15a..b26a7d5 100644 --- a/src/main/java/top/penowl/quidproquo/Ritual.java +++ b/src/main/java/top/penowl/quidproquo/Ritual.java @@ -10,22 +10,42 @@ import org.bukkit.entity.Player; import org.bukkit.inventory.ItemStack; public abstract class Ritual { + + // ingredients, map of type to count public HashMap ingredients = new HashMap(); + + // mob sacrifices, map of type to count public HashMap sacrifices = new HashMap(); + + // extra byproducts, prob shouldn't need to be used public ArrayList byproducts = new ArrayList(); + + // the amount of heath that is drained from doing the sacrifice public int health = 0; + + // chance of it backfireing and swapping the target and caster, ranges from 0.0 to 1.0, should be 0.0 for rituals with no target public double backfire = 0.0; + + // name of ritual, should be a gerund public String name = "unnamed"; + // whether you want a lightning effect + public Boolean lightning = false; + + // the actual effect that gets triggered public abstract void execute(Player caster, Player target, Location location); + // add an ingredient to the recipe (type and count) public void addIngredient(Material material, int count) { ingredients.put(material, count); } + // add a sacrifice to the recipe (type and count) public void addSacrifice(EntityType type, int count) { sacrifices.put(type, count); } + // this is where you set up the recipe, e.g. ingredients, sacrifices, lightning, health, name, etc public abstract void setup(); + } \ No newline at end of file