diff --git a/BauSystem_12/src/de/steamwar/bausystem/world/TNTSimulator_12.java b/BauSystem_12/src/de/steamwar/bausystem/world/TNTSimulator_12.java new file mode 100644 index 0000000..099c96f --- /dev/null +++ b/BauSystem_12/src/de/steamwar/bausystem/world/TNTSimulator_12.java @@ -0,0 +1,34 @@ +/* + * This file is a part of the SteamWar software. + * + * Copyright (C) 2020 SteamWar.de-Serverteam + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU Affero General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU Affero General Public License for more details. + * + * You should have received a copy of the GNU Affero General Public License + * along with this program. If not, see . + */ + +package de.steamwar.bausystem.world; + +import org.bukkit.Material; + +public class TNTSimulator_12 { + + public static Material active() { + return Material.CONCRETE; + } + + public static Material notActive() { + return Material.CONCRETE; + } + +} diff --git a/BauSystem_15/src/de/steamwar/bausystem/world/TNTSimulator_15.java b/BauSystem_15/src/de/steamwar/bausystem/world/TNTSimulator_15.java new file mode 100644 index 0000000..8fea70d --- /dev/null +++ b/BauSystem_15/src/de/steamwar/bausystem/world/TNTSimulator_15.java @@ -0,0 +1,35 @@ +/* + * + * This file is a part of the SteamWar software. + * + * Copyright (C) 2020 SteamWar.de-Serverteam + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU Affero General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU Affero General Public License for more details. + * + * You should have received a copy of the GNU Affero General Public License + * along with this program. If not, see . + */ + +package de.steamwar.bausystem.world; + +import org.bukkit.Material; + +public class TNTSimulator_15 { + + public static Material active() { + return Material.LIME_CONCRETE; + } + + public static Material notActive() { + return Material.RED_CONCRETE; + } + +} diff --git a/BauSystem_Main/src/de/steamwar/bausystem/BauSystem.java b/BauSystem_Main/src/de/steamwar/bausystem/BauSystem.java index b96ad37..c388e29 100644 --- a/BauSystem_Main/src/de/steamwar/bausystem/BauSystem.java +++ b/BauSystem_Main/src/de/steamwar/bausystem/BauSystem.java @@ -97,6 +97,8 @@ public class BauSystem extends JavaPlugin implements Listener { getCommand("detonator").setExecutor(new CommandDetonator()); getCommand("detonator").setTabCompleter(new CommandDetonatorTabCompleter()); getCommand("script").setExecutor(new CommandScript()); + getCommand("simulator").setExecutor(new CommandSimulator()); + getCommand("simulator").setTabCompleter(new CommandSimulatorTabCompleter()); getCommand("gui").setExecutor(new CommandGUI()); Bukkit.getPluginManager().registerEvents(this, this); @@ -104,6 +106,7 @@ public class BauSystem extends JavaPlugin implements Listener { Bukkit.getPluginManager().registerEvents(new ScriptListener(), this); Bukkit.getPluginManager().registerEvents(new BauScoreboard(), this); Bukkit.getPluginManager().registerEvents(new ClipboardListener(), this); + Bukkit.getPluginManager().registerEvents(new TNTSimulatorListener(), this); Bukkit.getPluginManager().registerEvents(new CommandGUI(), this); new AFKStopper(); diff --git a/BauSystem_Main/src/de/steamwar/bausystem/commands/CommandGUI.java b/BauSystem_Main/src/de/steamwar/bausystem/commands/CommandGUI.java index be01a34..5576d67 100644 --- a/BauSystem_Main/src/de/steamwar/bausystem/commands/CommandGUI.java +++ b/BauSystem_Main/src/de/steamwar/bausystem/commands/CommandGUI.java @@ -55,8 +55,6 @@ public class CommandGUI implements CommandExecutor, Listener { inv.setItem(37, getMaterial("GLASS_PANE", "THIN_GLASS"), "§7Platzhalter", clickType -> { }); - inv.setItem(38, getMaterial("GLASS_PANE", "THIN_GLASS"), "§7Platzhalter", clickType -> { - }); inv.setItem(43, getMaterial("GLASS_PANE", "THIN_GLASS"), "§7Platzhalter", clickType -> { }); inv.setItem(42, Material.NETHER_STAR, "§7Bau GUI Item", Arrays.asList("§7Du kannst dieses Item zum Öffnen der BauGUI nutzen", "§7oder Doppel F (Swap hands) drücken."), false, clickType -> { @@ -64,14 +62,7 @@ public class CommandGUI implements CommandExecutor, Listener { player.performCommand("gui item"); }); - ItemStack dtWand = Detonator.WAND.clone(); - ItemMeta meta = dtWand.getItemMeta(); - List lore = meta.getLore(); - lore.add("§8/§7dt wand"); - if (Welt.noPermission(player, Permission.world)) - lore.add("§cDu hast keine Worldrechte"); - meta.setLore(lore); - dtWand.setItemMeta(meta); + ItemStack dtWand = wand(player, Detonator.WAND, "§8/§7dt wand", Permission.world, "§cDu hast keine Worldrechte"); inv.setItem(39, dtWand, clickType -> { if (Welt.noPermission(player, Permission.world)) return; @@ -79,6 +70,14 @@ public class CommandGUI implements CommandExecutor, Listener { player.performCommand("dt wand"); }); + ItemStack simWand = wand(player, TNTSimulator.WAND, "§8/§7sim wand", Permission.world, "§cDu hast keine Worldrechte"); + inv.setItem(38, simWand, clickType -> { + if (Welt.noPermission(player, Permission.world)) + return; + player.closeInventory(); + player.performCommand("sim wand"); + }); + inv.setItem(40, getMaterial("WOODEN_AXE", "WOOD_AXE"), "§eWorldedit Axt", getNoPermsLore(Arrays.asList("§8//§7wand"), player, "§cDu hast keine Worldeditrechte", Permission.worldedit), false, clickType -> { if (Welt.noPermission(player, Permission.world)) return; @@ -567,6 +566,18 @@ public class CommandGUI implements CommandExecutor, Listener { return null; } + private static ItemStack wand(Player player, ItemStack base, String command, Permission permission, String noPermissionMessage) { + base = base.clone(); + ItemMeta meta = base.getItemMeta(); + List lore = meta.getLore(); + lore.add(command); + if (Welt.noPermission(player, permission)) + lore.add(noPermissionMessage); + meta.setLore(lore); + base.setItemMeta(meta); + return base; + } + @Override public boolean onCommand(CommandSender commandSender, Command command, String s, String[] strings) { if (!(commandSender instanceof Player)) diff --git a/BauSystem_Main/src/de/steamwar/bausystem/commands/CommandSimulator.java b/BauSystem_Main/src/de/steamwar/bausystem/commands/CommandSimulator.java new file mode 100644 index 0000000..325f1a3 --- /dev/null +++ b/BauSystem_Main/src/de/steamwar/bausystem/commands/CommandSimulator.java @@ -0,0 +1,79 @@ +/* + * + * This file is a part of the SteamWar software. + * + * Copyright (C) 2020 SteamWar.de-Serverteam + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU Affero General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU Affero General Public License for more details. + * + * You should have received a copy of the GNU Affero General Public License + * along with this program. If not, see . + */ + +package de.steamwar.bausystem.commands; + +import de.steamwar.bausystem.BauSystem; +import de.steamwar.bausystem.Permission; +import de.steamwar.bausystem.PlayerUtils; +import de.steamwar.bausystem.world.TNTSimulator; +import de.steamwar.bausystem.world.Welt; +import org.bukkit.command.Command; +import org.bukkit.command.CommandExecutor; +import org.bukkit.command.CommandSender; +import org.bukkit.entity.Player; + +public class CommandSimulator implements CommandExecutor { + + private void help(Player player) { + player.sendMessage("§8/§esimulator §8- §7Öffnet die Simulations GUI"); + player.sendMessage("§8/§esimulator start §8- §7Startet die Simulation"); + player.sendMessage("§8/§esimulator wand §8- §7Legt dir den Simulatorstab ins Inventar"); + player.sendMessage("§8/§esimulator delete §8- §7Löscht alle TNT"); + } + + private boolean permissionCheck(Player player) { + if (Welt.noPermission(player, Permission.world)) { + player.sendMessage(BauSystem.PREFIX + "§cDu darfst hier nicht den Simulator nutzen"); + return false; + } + return true; + } + + @Override + public boolean onCommand(CommandSender commandSender, Command command, String s, String[] args) { + if (!(commandSender instanceof Player)) + return false; + Player player = (Player) commandSender; + if (!permissionCheck(player)) { + return false; + } + + if (args.length == 1) { + switch (args[0].toLowerCase()) { + case "wand": + PlayerUtils.giveItemToPlayer(player, TNTSimulator.WAND); + break; + case "start": + TNTSimulator.get(player).start(); + break; + case "delete": + TNTSimulator.get(player).delete(); + default: + help(player); + break; + } + return false; + } + TNTSimulator.openSimulator(player); + return false; + } + +} diff --git a/BauSystem_Main/src/de/steamwar/bausystem/commands/CommandSimulatorTabCompleter.java b/BauSystem_Main/src/de/steamwar/bausystem/commands/CommandSimulatorTabCompleter.java new file mode 100644 index 0000000..0103d30 --- /dev/null +++ b/BauSystem_Main/src/de/steamwar/bausystem/commands/CommandSimulatorTabCompleter.java @@ -0,0 +1,60 @@ +/* + * + * This file is a part of the SteamWar software. + * + * Copyright (C) 2020 SteamWar.de-Serverteam + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU Affero General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU Affero General Public License for more details. + * + * You should have received a copy of the GNU Affero General Public License + * along with this program. If not, see . + */ + +package de.steamwar.bausystem.commands; + +import org.bukkit.command.Command; +import org.bukkit.command.CommandSender; +import org.bukkit.command.TabCompleter; +import org.bukkit.entity.Player; + +import java.util.ArrayList; +import java.util.List; + +public class CommandSimulatorTabCompleter implements TabCompleter { + + @Override + public List onTabComplete(CommandSender sender, Command command, String label, String[] args) { + if(!(sender instanceof Player)) return new ArrayList<>(); + return simulatorTabComplete((Player) sender, args); + } + + private List simulatorTabComplete(Player player, String[] args) { + List tabComplete = new ArrayList<>(); + tabComplete.add("wand"); + tabComplete.add("start"); + tabComplete.add("delete"); + + if (args.length >= 2) { + return new ArrayList<>(); + } + return manageList(tabComplete, args, 0); + } + + private List manageList(List strings, String[] args, int index) { + for (int i = strings.size() - 1; i >= 0; i--) { + if (!strings.get(i).startsWith(args[index])) { + strings.remove(i); + } + } + return strings; + } + +} diff --git a/BauSystem_Main/src/de/steamwar/bausystem/world/TNTSimulator.java b/BauSystem_Main/src/de/steamwar/bausystem/world/TNTSimulator.java new file mode 100644 index 0000000..7a9fb2d --- /dev/null +++ b/BauSystem_Main/src/de/steamwar/bausystem/world/TNTSimulator.java @@ -0,0 +1,441 @@ +/* + * + * This file is a part of the SteamWar software. + * + * Copyright (C) 2020 SteamWar.de-Serverteam + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU Affero General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU Affero General Public License for more details. + * + * You should have received a copy of the GNU Affero General Public License + * along with this program. If not, see . + * / + */ + +package de.steamwar.bausystem.world; + +import de.steamwar.bausystem.BauSystem; +import de.steamwar.core.VersionedCallable; +import de.steamwar.inventory.SWAnvilInv; +import de.steamwar.inventory.SWInventory; +import de.steamwar.inventory.SWItem; +import de.steamwar.inventory.SWListInv; +import org.bukkit.Bukkit; +import org.bukkit.Material; +import org.bukkit.World; +import org.bukkit.entity.Player; +import org.bukkit.entity.TNTPrimed; +import org.bukkit.inventory.ItemStack; +import org.bukkit.util.Consumer; +import org.bukkit.util.Vector; + +import java.util.*; +import java.util.concurrent.atomic.AtomicInteger; + +public class TNTSimulator { + + private static final Vector X_VECTOR = new Vector(0.0625, 0, 0); + private static final Vector NX_VECTOR = new Vector(-0.0625, 0, 0); + private static final Vector Y_VECTOR = new Vector(0, 0.0625, 0); + private static final Vector NY_VECTOR = new Vector(0, -0.0625, 0); + private static final Vector Z_VECTOR = new Vector(0, 0, 0.0625); + private static final Vector NZ_VECTOR = new Vector(0, 0, -0.0625); + private static final List LORE = Collections.singletonList("§eZum ändern klicken"); + private static final List EMPTY = new ArrayList<>(); + + static final Map TNT_SIMULATOR_MAP = new HashMap<>(); + private final Set TNT_SPAWNS = new HashSet<>(); + + public static final ItemStack WAND = new SWItem(Material.BLAZE_ROD, "§eKanonensimulator", Arrays.asList("§eRechtsklick Block §8- §7Füge einen TNT hinzu", "§eRechtsklick Luft §8- §7Öffne den Simulator", "§eLinksklick §8- §7Starte die Simulation"), false, null).getItemStack(); + + public static TNTSimulator get(Player player) { + return TNT_SIMULATOR_MAP.computeIfAbsent(player, p -> new TNTSimulator()); + } + + public static void openSimulator(Player player) { + TNTSimulator tntSimulator = get(player); + + List> swListEntryList = new ArrayList<>(); + tntSimulator.TNT_SPAWNS.forEach(tntSpawn -> { + List lore = new ArrayList<>(); + lore.add("§7Klicken zum Konfigurieren"); + lore.add(""); + lore.add("§7TNT-Anzahl§8: §e" + tntSpawn.getCount()); + lore.add("§7Tick§8: §e" + tntSpawn.getTickOffset()); + lore.add("§7Fuse-Tick§8: §e" + tntSpawn.getFuseTicks()); + lore.add(""); + lore.add("§7X§8: §e" + tntSpawn.getPosition().getX()); + lore.add("§7Y§8: §e" + tntSpawn.getPosition().getY()); + lore.add("§7Z§8: §e" + tntSpawn.getPosition().getZ()); + swListEntryList.add(new SWListInv.SWListEntry<>(new SWItem(Material.TNT, "§eTNT", lore, false, null), tntSpawn)); + }); + swListEntryList.sort(Comparator.comparing(SWListInv.SWListEntry::getObject)); + + SWListInv swListInv = new SWListInv<>(player, "Kanonensimulator", false, swListEntryList, (clickType, tntSpawn) -> { + editTNT(player, tntSpawn); + }); + swListInv.setItem(51, new SWItem(Material.BARRIER, "§cTNT löschen", clickType -> { + tntSimulator.TNT_SPAWNS.clear(); + openSimulator(player); + })); + swListInv.setItem(47, new SWItem(Material.FLINT_AND_STEEL, "§eStarten", clickType -> { + player.closeInventory(); + startSimulation(player); + })); + swListInv.open(); + } + + static void editTNT(Player player, TNTSpawn tntSpawn) { + TNTSimulator tntSimulator = get(player); + + SWInventory swInventory = new SWInventory(player, 54, "TNT"); + swInventory.setItem(49, new SWItem(Material.REDSTONE_BLOCK, "§cZurück", clickType -> { + openSimulator(player); + })); + + // Delete tnt + swInventory.setItem(53, new SWItem(Material.BARRIER, "§cEntfernen", clickType -> { + tntSimulator.TNT_SPAWNS.remove(tntSpawn); + openSimulator(player); + })); + + // Change Count of spawned TNT + swInventory.setItem(10, new SWItem(SWItem.getDye(10), "§7+1", clickType -> { + tntSpawn.setCount(tntSpawn.getCount() + 1); + if (tntSpawn.getCount() > 400) { + tntSpawn.setCount(400); + } + editTNT(player, tntSpawn); + })); + swInventory.setItem(19, new SWItem(Material.TNT, "§7Anzahl §8- §e" + tntSpawn.getCount(), LORE, false, clickType -> { + changeCount(player, "Anzahl TNT", tntSpawn.getCount(), count -> { + if (count < 1) count = 1; + if (count > 400) count = 400; + tntSpawn.setCount(count); + editTNT(player, tntSpawn); + }, () -> editTNT(player, tntSpawn)); + })); + swInventory.setItem(28, new SWItem(SWItem.getDye(1), "§7-1", clickType -> { + tntSpawn.setCount(tntSpawn.getCount() - 1); + if (tntSpawn.getCount() < 1) { + tntSpawn.setCount(1); + } + editTNT(player, tntSpawn); + })); + + // Change TickOffset + swInventory.setItem(11, new SWItem(SWItem.getDye(10), "§7+1", clickType -> { + tntSpawn.setTickOffset(tntSpawn.getTickOffset() + 1); + if (tntSpawn.getTickOffset() > 8000) { + tntSpawn.setTickOffset(8000); + } + editTNT(player, tntSpawn); + })); + swInventory.setItem(20, new SWItem(Material.CLOCK, "§7Tick §8- §e" + tntSpawn.getTickOffset(), LORE, false, clickType -> { + changeCount(player, "Tick Offset", tntSpawn.getTickOffset(), tick -> { + if (tick < 0) tick = 0; + if (tick > 8000) tick = 8000; + tntSpawn.setTickOffset(tick); + editTNT(player, tntSpawn); + }, () -> editTNT(player, tntSpawn)); + })); + swInventory.setItem(29, new SWItem(SWItem.getDye(1), "§7-1", clickType -> { + tntSpawn.setTickOffset(tntSpawn.getTickOffset() - 1); + if (tntSpawn.getTickOffset() < 0) { + tntSpawn.setTickOffset(0); + } + editTNT(player, tntSpawn); + })); + + // Change FuseTicks + swInventory.setItem(12, new SWItem(SWItem.getDye(10), "§7+1", clickType -> { + tntSpawn.setFuseTicks(tntSpawn.getFuseTicks() + 1); + if (tntSpawn.getFuseTicks() > 80) { + tntSpawn.setFuseTicks(80); + } + editTNT(player, tntSpawn); + })); + swInventory.setItem(21, new SWItem(Material.CLOCK, "§7Fuse-Ticks §8- §e" + tntSpawn.getFuseTicks(), LORE, false, clickType -> { + changeCount(player, "Fuse-Ticks", tntSpawn.getFuseTicks(), tick -> { + if (tick < 0) tick = 0; + if (tick > 80) tick = 80; + tntSpawn.setFuseTicks(tick); + editTNT(player, tntSpawn); + }, () -> editTNT(player, tntSpawn)); + })); + swInventory.setItem(30, new SWItem(SWItem.getDye(1), "§7-1", clickType -> { + tntSpawn.setFuseTicks(tntSpawn.getFuseTicks() - 1); + if (tntSpawn.getFuseTicks() < 1) { + tntSpawn.setFuseTicks(1); + } + editTNT(player, tntSpawn); + })); + + // Repeater before Comparator + swInventory.setItem(4, new SWItem(comparatorOrNot(tntSpawn.isComparator()), "§7Gezündet durch §8- §e" + (tntSpawn.isComparator() ? "Comparator" : "Repeater"), clickType -> { + tntSpawn.setComparator(!tntSpawn.isComparator()); + editTNT(player, tntSpawn); + })); + + // Velocity Settings + swInventory.setItem(13, new SWItem(activeOrNot(tntSpawn.isxVelocity()), "§7TNT §eSprung X §8- " + active(tntSpawn.isxVelocity()), clickType -> { + tntSpawn.setxVelocity(!tntSpawn.isxVelocity()); + editTNT(player, tntSpawn); + })); + swInventory.setItem(22, new SWItem(activeOrNot(tntSpawn.isyVelocity()), "§7TNT §eSprung Y §8- " + active(tntSpawn.isyVelocity()), clickType -> { + tntSpawn.setyVelocity(!tntSpawn.isyVelocity()); + editTNT(player, tntSpawn); + })); + swInventory.setItem(31, new SWItem(activeOrNot(tntSpawn.iszVelocity()), "§7TNT §eSprung Z §8- " + active(tntSpawn.iszVelocity()), clickType -> { + tntSpawn.setzVelocity(!tntSpawn.iszVelocity()); + editTNT(player, tntSpawn); + })); + + // X Position + swInventory.setItem(14, new SWItem(SWItem.getDye(10), "§7+0,0625", clickType -> { + tntSpawn.getPosition().add(X_VECTOR); + editTNT(player, tntSpawn); + })); + swInventory.setItem(23, new SWItem(Material.PAPER, "§7x-Position §8- §e" + tntSpawn.getPosition().getX(), LORE, false, clickType -> { + changePosition(player, tntSpawn.getPosition().getX(), x -> { + tntSpawn.getPosition().setX(clamp(x)); + editTNT(player, tntSpawn); + }, () -> editTNT(player, tntSpawn)); + })); + swInventory.setItem(32, new SWItem(SWItem.getDye(1), "§7-0,0625", clickType -> { + tntSpawn.getPosition().add(NX_VECTOR); + editTNT(player, tntSpawn); + })); + + // Y Position + swInventory.setItem(15, new SWItem(SWItem.getDye(10), "§7+0,0625", clickType -> { + tntSpawn.getPosition().add(Y_VECTOR); + editTNT(player, tntSpawn); + })); + swInventory.setItem(24, new SWItem(Material.PAPER, "§7y-Position §8- §e" + tntSpawn.getPosition().getY(), LORE, false, clickType -> { + changePosition(player, tntSpawn.getPosition().getY(), y -> { + tntSpawn.getPosition().setY(clamp(y)); + editTNT(player, tntSpawn); + }, () -> editTNT(player, tntSpawn)); + })); + swInventory.setItem(33, new SWItem(SWItem.getDye(1), "§7-0,0625", clickType -> { + tntSpawn.getPosition().add(NY_VECTOR); + editTNT(player, tntSpawn); + })); + + // Z Position + swInventory.setItem(16, new SWItem(SWItem.getDye(10), "§7+0,0625", clickType -> { + tntSpawn.getPosition().add(Z_VECTOR); + editTNT(player, tntSpawn); + })); + swInventory.setItem(25, new SWItem(Material.PAPER, "§7z-Position §8- §e" + tntSpawn.getPosition().getZ(), LORE, false, clickType -> { + changePosition(player, tntSpawn.getPosition().getZ(), z -> { + tntSpawn.getPosition().setZ(clamp(z)); + editTNT(player, tntSpawn); + }, () -> editTNT(player, tntSpawn)); + })); + swInventory.setItem(34, new SWItem(SWItem.getDye(1), "§7-0,0625", clickType -> { + tntSpawn.getPosition().add(NZ_VECTOR); + editTNT(player, tntSpawn); + })); + swInventory.open(); + } + + static void startSimulation(Player player) { + TNT_SIMULATOR_MAP.getOrDefault(player, new TNTSimulator()).start(); + } + + static void addTNT(Player player, TNTSpawn tntSpawn) { + TNTSimulator tntSimulator = TNT_SIMULATOR_MAP.computeIfAbsent(player, player1 -> new TNTSimulator()); + tntSimulator.TNT_SPAWNS.add(tntSpawn); + } + + private static Material comparatorOrNot(boolean b) { + if (b) { + return SWItem.getMaterial("REDSTONE_COMPARATOR"); + } else { + return SWItem.getMaterial("DIODE"); + } + } + + private static Material activeOrNot(boolean b) { + if (b) { + return VersionedCallable.call(new VersionedCallable<>(TNTSimulator_12::active, 8), new VersionedCallable<>(TNTSimulator_15::active, 14)); + } else { + return VersionedCallable.call(new VersionedCallable<>(TNTSimulator_12::notActive, 8), new VersionedCallable<>(TNTSimulator_15::notActive, 14)); + } + } + + private static String active(boolean b) { + return b ? "§aan" : "§caus"; + } + + private static void changeCount(Player player, String name, int defaultValue, Consumer result, Runnable failure) { + SWAnvilInv swAnvilInv = new SWAnvilInv(player, name, defaultValue + ""); + swAnvilInv.setItem(Material.PAPER); + swAnvilInv.setCallback(s -> { + try { + result.accept(Integer.parseInt(s)); + } catch (NumberFormatException e) { + failure.run(); + } + }); + swAnvilInv.open(); + } + + private static void changePosition(Player player, double defaultValue, Consumer result, Runnable failure) { + SWAnvilInv swAnvilInv = new SWAnvilInv(player, "Position", defaultValue + ""); + swAnvilInv.setItem(Material.PAPER); + swAnvilInv.setCallback(s -> { + try { + result.accept(Double.parseDouble(s.replace(',', '.'))); + } catch (NumberFormatException e) { + failure.run(); + } + }); + swAnvilInv.open(); + } + + private static double clamp(double d) { + return (int)(d * 10000) * 0.0001; + } + + public void delete() { + TNT_SPAWNS.clear(); + } + + public void start() { + Map> first = new HashMap<>(); + Map> second = new HashMap<>(); + int lastTick = 0; + + for (TNTSpawn tntSpawn : TNT_SPAWNS) { + Map> list = tntSpawn.isComparator() ? second : first; + for (int i = 0; i < tntSpawn.getCount(); i++) { + list.computeIfAbsent(tntSpawn.getTickOffset(), integer -> new ArrayList<>()).add(tntSpawn); + } + if (lastTick < tntSpawn.getTickOffset()) { + lastTick = tntSpawn.getTickOffset(); + } + } + int finalLastTick = lastTick; + + AtomicInteger currentTick = new AtomicInteger(0); + Bukkit.getScheduler().runTaskTimer(BauSystem.getPlugin(), bukkitTask -> { + int tick = currentTick.get(); + spawnRandomList(first.getOrDefault(tick, EMPTY)); + spawnRandomList(second.getOrDefault(tick, EMPTY)); + if (tick > finalLastTick) bukkitTask.cancel(); + currentTick.incrementAndGet(); + }, 1, 1); + } + + private void spawnRandomList(List tntSpawns) { + if (tntSpawns.isEmpty()) return; + Collections.shuffle(tntSpawns); + for (TNTSpawn tntSpawn : tntSpawns) { + tntSpawn.spawn(); + } + } + + public static class TNTSpawn implements Comparable { + + private static final World WORLD = Bukkit.getWorlds().get(0); + + private final Vector position; + private int fuseTicks = 80; + private int count = 1; + private int tickOffset = 0; + private boolean xVelocity = true; + private boolean yVelocity = true; + private boolean zVelocity = true; + private boolean comparator = false; + + public TNTSpawn(Vector position) { + this.position = position; + } + + public void spawn() { + WORLD.spawn(position.toLocation(WORLD), TNTPrimed.class, tntPrimed -> { + tntPrimed.setFuseTicks(fuseTicks); + if (!xVelocity) tntPrimed.setVelocity(tntPrimed.getVelocity().setX(0)); + if (!yVelocity) tntPrimed.setVelocity(tntPrimed.getVelocity().setY(0)); + if (!zVelocity) tntPrimed.setVelocity(tntPrimed.getVelocity().setZ(0)); + }); + } + + public Vector getPosition() { + return position; + } + + public int getFuseTicks() { + return fuseTicks; + } + + public void setFuseTicks(int fuseTicks) { + this.fuseTicks = fuseTicks; + } + + public int getCount() { + return count; + } + + public void setCount(int count) { + this.count = count; + } + + public int getTickOffset() { + return tickOffset; + } + + public void setTickOffset(int tickOffset) { + this.tickOffset = tickOffset; + } + + public boolean isxVelocity() { + return xVelocity; + } + + public void setxVelocity(boolean xVelocity) { + this.xVelocity = xVelocity; + } + + public boolean isyVelocity() { + return yVelocity; + } + + public void setyVelocity(boolean yVelocity) { + this.yVelocity = yVelocity; + } + + public boolean iszVelocity() { + return zVelocity; + } + + public void setzVelocity(boolean zVelocity) { + this.zVelocity = zVelocity; + } + + public boolean isComparator() { + return comparator; + } + + public void setComparator(boolean comparator) { + this.comparator = comparator; + } + + @Override + public int compareTo(TNTSpawn tntSpawn) { + return -Integer.compare(tickOffset, tntSpawn.tickOffset); + } + + } + +} diff --git a/BauSystem_Main/src/de/steamwar/bausystem/world/TNTSimulatorListener.java b/BauSystem_Main/src/de/steamwar/bausystem/world/TNTSimulatorListener.java new file mode 100644 index 0000000..e0af211 --- /dev/null +++ b/BauSystem_Main/src/de/steamwar/bausystem/world/TNTSimulatorListener.java @@ -0,0 +1,84 @@ +/* + * + * This file is a part of the SteamWar software. + * + * Copyright (C) 2020 SteamWar.de-Serverteam + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU Affero General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU Affero General Public License for more details. + * + * You should have received a copy of the GNU Affero General Public License + * along with this program. If not, see . + * / + */ + +package de.steamwar.bausystem.world; + +import de.steamwar.bausystem.BauSystem; +import de.steamwar.bausystem.Permission; +import org.bukkit.entity.Player; +import org.bukkit.event.EventHandler; +import org.bukkit.event.Listener; +import org.bukkit.event.player.PlayerInteractEvent; +import org.bukkit.event.player.PlayerQuitEvent; +import org.bukkit.util.Vector; + +import static de.steamwar.bausystem.world.TNTSimulator.*; + +public class TNTSimulatorListener implements Listener { + + private static final Vector HALF = new Vector(0.5, 0, 0.5); + + private boolean permissionCheck(Player player) { + if (Welt.noPermission(player, Permission.world)) { + player.sendMessage(BauSystem.PREFIX + "§cDu darfst hier nicht den Simulator nutzen"); + return false; + } + return true; + } + + @EventHandler + public void onPlayerInteract(PlayerInteractEvent event) { + if (event.getItem() == null) { + return; + } + if (!event.getItem().isSimilar(WAND)) { + return; + } + event.setCancelled(true); + if (!permissionCheck(event.getPlayer())) { + return; + } + + switch (event.getAction()) { + case LEFT_CLICK_BLOCK: + case LEFT_CLICK_AIR: + startSimulation(event.getPlayer()); + break; + case RIGHT_CLICK_BLOCK: + Vector location = event.getClickedBlock().getLocation().toVector().add(event.getBlockFace().getDirection()).add(HALF); + TNTSpawn tntSpawn = new TNTSpawn(location); + addTNT(event.getPlayer(), tntSpawn); + editTNT(event.getPlayer(), tntSpawn); + break; + case RIGHT_CLICK_AIR: + openSimulator(event.getPlayer()); + break; + default: + break; + } + } + + @EventHandler + public void onPlayerQuit(PlayerQuitEvent event) { + TNT_SIMULATOR_MAP.remove(event.getPlayer()); + } + +} diff --git a/BauSystem_Main/src/plugin.yml b/BauSystem_Main/src/plugin.yml index 8ac3fdc..113cb9a 100644 --- a/BauSystem_Main/src/plugin.yml +++ b/BauSystem_Main/src/plugin.yml @@ -36,4 +36,6 @@ commands: detonator: aliases: dt script: + simulator: + aliases: sim gui: \ No newline at end of file