Compare commits
17 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
1a75a7db33 | ||
|
|
be78c34222 | ||
|
|
2460691df2 | ||
|
|
46dc0cbe05 | ||
|
|
79fed93d20 | ||
|
|
c3a8f15f6d | ||
|
|
b46af01475 | ||
|
|
8fe8f2b1af | ||
|
|
ce6847e58e | ||
|
|
95b0e2f971 | ||
|
|
045a5f2b1b | ||
| 8a2c3b4277 | |||
| 38bbba4c8b | |||
| d92d0d5328 | |||
| a27c174daa | |||
| 0dddc0dfb5 | |||
| 7fe5bd62c7 |
2
pom.xml
2
pom.xml
@@ -6,7 +6,7 @@
|
|||||||
|
|
||||||
<groupId>de.nexuslobby</groupId>
|
<groupId>de.nexuslobby</groupId>
|
||||||
<artifactId>NexusLobby</artifactId>
|
<artifactId>NexusLobby</artifactId>
|
||||||
<version>1.1.3</version>
|
<version>1.9</version>
|
||||||
<packaging>jar</packaging>
|
<packaging>jar</packaging>
|
||||||
|
|
||||||
<name>NexusLobby</name>
|
<name>NexusLobby</name>
|
||||||
|
|||||||
@@ -204,18 +204,55 @@ public class NexusLobby extends JavaPlugin implements Listener {
|
|||||||
|
|
||||||
private void checkUpdates() {
|
private void checkUpdates() {
|
||||||
new UpdateChecker(this).getVersion(version -> {
|
new UpdateChecker(this).getVersion(version -> {
|
||||||
if (!this.getDescription().getVersion().equalsIgnoreCase(version)) {
|
String currentVersion = this.getDescription().getVersion();
|
||||||
|
if (isRemoteVersionNewer(currentVersion, version)) {
|
||||||
this.updateAvailable = true;
|
this.updateAvailable = true;
|
||||||
this.latestVersion = version;
|
this.latestVersion = version;
|
||||||
getLogger().warning("====================================================");
|
getLogger().warning("====================================================");
|
||||||
getLogger().warning("Update gefunden! v" + getDescription().getVersion() + " -> v" + version);
|
getLogger().warning("Update gefunden! v" + currentVersion + " -> v" + version);
|
||||||
getLogger().warning("====================================================");
|
getLogger().warning("====================================================");
|
||||||
} else {
|
} else {
|
||||||
getLogger().info("NexusLobby ist aktuell (v" + version + ").");
|
this.updateAvailable = false;
|
||||||
|
this.latestVersion = "";
|
||||||
|
getLogger().info("NexusLobby ist aktuell/neu genug (lokal v" + currentVersion + ", remote v" + version + ").");
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private boolean isRemoteVersionNewer(String localVersion, String remoteVersion) {
|
||||||
|
int[] local = extractVersionNumbers(localVersion);
|
||||||
|
int[] remote = extractVersionNumbers(remoteVersion);
|
||||||
|
|
||||||
|
int max = Math.max(local.length, remote.length);
|
||||||
|
for (int i = 0; i < max; i++) {
|
||||||
|
int l = i < local.length ? local[i] : 0;
|
||||||
|
int r = i < remote.length ? remote[i] : 0;
|
||||||
|
if (r > l) return true;
|
||||||
|
if (r < l) return false;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private int[] extractVersionNumbers(String version) {
|
||||||
|
if (version == null || version.isBlank()) return new int[0];
|
||||||
|
|
||||||
|
java.util.regex.Matcher matcher = java.util.regex.Pattern.compile("\\\\d+").matcher(version);
|
||||||
|
java.util.List<Integer> parts = new java.util.ArrayList<>();
|
||||||
|
while (matcher.find()) {
|
||||||
|
try {
|
||||||
|
parts.add(Integer.parseInt(matcher.group()));
|
||||||
|
} catch (NumberFormatException ignored) {
|
||||||
|
parts.add(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int[] result = new int[parts.size()];
|
||||||
|
for (int i = 0; i < parts.size(); i++) {
|
||||||
|
result[i] = parts.get(i);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
private void registerModules() {
|
private void registerModules() {
|
||||||
moduleManager.registerModule(new ProtectionModule());
|
moduleManager.registerModule(new ProtectionModule());
|
||||||
moduleManager.registerModule(new ScoreboardModule());
|
moduleManager.registerModule(new ScoreboardModule());
|
||||||
@@ -323,7 +360,7 @@ public class NexusLobby extends JavaPlugin implements Listener {
|
|||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("update_available"));
|
player.sendMessage(de.nexuslobby.utils.LangManager.get("update_available"));
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("update_version").replace("{old}", getDescription().getVersion()).replace("{new}", latestVersion));
|
player.sendMessage(de.nexuslobby.utils.LangManager.get("update_version").replace("{old}", getDescription().getVersion()).replace("{new}", latestVersion));
|
||||||
TextComponent link = new TextComponent(de.nexuslobby.utils.LangManager.get("update_download_link"));
|
TextComponent link = new TextComponent(de.nexuslobby.utils.LangManager.get("update_download_link"));
|
||||||
link.setClickEvent(new ClickEvent(ClickEvent.Action.OPEN_URL, "https://git.viper.ipv64.net/M_Viper/NexusLobby/releases"));
|
link.setClickEvent(new ClickEvent(ClickEvent.Action.OPEN_URL, "https://www.spigotmc.org/resources/132388/"));
|
||||||
link.setHoverEvent(new HoverEvent(HoverEvent.Action.SHOW_TEXT,
|
link.setHoverEvent(new HoverEvent(HoverEvent.Action.SHOW_TEXT,
|
||||||
new ComponentBuilder(de.nexuslobby.utils.LangManager.get("update_download_hover")).create()));
|
new ComponentBuilder(de.nexuslobby.utils.LangManager.get("update_download_hover")).create()));
|
||||||
player.spigot().sendMessage(link);
|
player.spigot().sendMessage(link);
|
||||||
@@ -492,6 +529,12 @@ public class NexusLobby extends JavaPlugin implements Listener {
|
|||||||
if (params.equalsIgnoreCase("parkour_top")) {
|
if (params.equalsIgnoreCase("parkour_top")) {
|
||||||
return parkourManager != null ? parkourManager.getTopTen() : "N/A";
|
return parkourManager != null ? parkourManager.getTopTen() : "N/A";
|
||||||
}
|
}
|
||||||
|
if (params.equalsIgnoreCase("parkour_top_1")) {
|
||||||
|
return parkourManager != null ? parkourManager.getTopTen(1) : "N/A";
|
||||||
|
}
|
||||||
|
if (params.equalsIgnoreCase("parkour_top_2")) {
|
||||||
|
return parkourManager != null ? parkourManager.getTopTen(2) : "N/A";
|
||||||
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ public class LobbyTabCompleter implements TabCompleter {
|
|||||||
private final HologramModule hologramModule;
|
private final HologramModule hologramModule;
|
||||||
|
|
||||||
public LobbyTabCompleter(PortalManager portalManager, HologramModule hologramModule) {
|
public LobbyTabCompleter(PortalManager portalManager, HologramModule hologramModule) {
|
||||||
this.portalManager = portalManager;
|
this.portalManager = portalManager;
|
||||||
this.hologramModule = hologramModule;
|
this.hologramModule = hologramModule;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -28,14 +28,14 @@ public class LobbyTabCompleter implements TabCompleter {
|
|||||||
List<String> suggestions = new ArrayList<>();
|
List<String> suggestions = new ArrayList<>();
|
||||||
String cmdName = command.getName().toLowerCase();
|
String cmdName = command.getName().toLowerCase();
|
||||||
|
|
||||||
// ── NexusLobby Hauptbefehl (/nexus oder /nexuslobby) ─────────────────
|
// ── NexusLobby Hauptbefehl (/nexus oder /nexuslobby) ──────────────────
|
||||||
if (cmdName.equals("nexuslobby") || cmdName.equals("nexus")) {
|
if (cmdName.equals("nexuslobby") || cmdName.equals("nexus")) {
|
||||||
|
|
||||||
// /nexuslobby <?>
|
// /nexuslobby <?>
|
||||||
if (args.length == 1) {
|
if (args.length == 1) {
|
||||||
if (sender.hasPermission("nexuslobby.admin")) {
|
if (sender.hasPermission("nexuslobby.admin")) {
|
||||||
suggestions.addAll(Arrays.asList(
|
suggestions.addAll(Arrays.asList(
|
||||||
"reload", "setspawn", "silentjoin", "parkour", "ball"
|
"reload", "setspawn", "silentjoin", "parkour", "ball", "cleanbubbles", "gadgetshield"
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
suggestions.add("sb");
|
suggestions.add("sb");
|
||||||
@@ -49,9 +49,17 @@ public class LobbyTabCompleter implements TabCompleter {
|
|||||||
suggestions.addAll(Arrays.asList("admin", "spieler"));
|
suggestions.addAll(Arrays.asList("admin", "spieler"));
|
||||||
}
|
}
|
||||||
case "silentjoin" -> suggestions.addAll(Arrays.asList("on", "off"));
|
case "silentjoin" -> suggestions.addAll(Arrays.asList("on", "off"));
|
||||||
case "parkour" -> suggestions.addAll(Arrays.asList(
|
case "parkour" -> {
|
||||||
"setstart", "setfinish", "setcheckpoint", "reset", "clear", "removeall"
|
if (sender.hasPermission("nexuslobby.admin")) {
|
||||||
));
|
suggestions.addAll(Arrays.asList(
|
||||||
|
"setstart", "setfinish", "setcheckpoint",
|
||||||
|
"holo", "reset", "clear", "removeall", "info"
|
||||||
|
));
|
||||||
|
} else {
|
||||||
|
// Normale Spieler können nur ihren Run abbrechen
|
||||||
|
suggestions.add("reset");
|
||||||
|
}
|
||||||
|
}
|
||||||
case "ball" -> {
|
case "ball" -> {
|
||||||
if (sender.hasPermission("nexuslobby.admin")) {
|
if (sender.hasPermission("nexuslobby.admin")) {
|
||||||
suggestions.addAll(Arrays.asList(
|
suggestions.addAll(Arrays.asList(
|
||||||
@@ -64,17 +72,21 @@ public class LobbyTabCompleter implements TabCompleter {
|
|||||||
// /nexuslobby <sub> <sub2> <?>
|
// /nexuslobby <sub> <sub2> <?>
|
||||||
} else if (args.length == 3) {
|
} else if (args.length == 3) {
|
||||||
switch (args[0].toLowerCase()) {
|
switch (args[0].toLowerCase()) {
|
||||||
|
// Für setstart / setfinish / setcheckpoint → Strecken-Nummer 1 oder 2
|
||||||
case "parkour" -> {
|
case "parkour" -> {
|
||||||
if (args[1].equalsIgnoreCase("setcheckpoint"))
|
String parkSub = args[1].toLowerCase();
|
||||||
suggestions.addAll(Arrays.asList("1","2","3","4","5","6","7","8","9"));
|
if (parkSub.equals("setstart")
|
||||||
|
|| parkSub.equals("setfinish")
|
||||||
|
|| parkSub.equals("setcheckpoint")
|
||||||
|
|| parkSub.equals("holo")) {
|
||||||
|
suggestions.addAll(Arrays.asList("1", "2"));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
case "ball" -> {
|
case "ball" -> {
|
||||||
switch (args[1].toLowerCase()) {
|
switch (args[1].toLowerCase()) {
|
||||||
// /nexuslobby ball goal <?>
|
case "goal" -> suggestions.addAll(Arrays.asList(
|
||||||
case "goal" -> suggestions.addAll(Arrays.asList(
|
|
||||||
"pos1", "pos2", "save", "delete", "list", "show", "debug"
|
"pos1", "pos2", "save", "delete", "list", "show", "debug"
|
||||||
));
|
));
|
||||||
// /nexuslobby ball score <?>
|
|
||||||
case "score" -> suggestions.add("reset");
|
case "score" -> suggestions.add("reset");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -84,19 +96,16 @@ public class LobbyTabCompleter implements TabCompleter {
|
|||||||
} else if (args.length == 4) {
|
} else if (args.length == 4) {
|
||||||
if (args[0].equalsIgnoreCase("ball") && args[1].equalsIgnoreCase("goal")) {
|
if (args[0].equalsIgnoreCase("ball") && args[1].equalsIgnoreCase("goal")) {
|
||||||
switch (args[2].toLowerCase()) {
|
switch (args[2].toLowerCase()) {
|
||||||
// /nexuslobby ball goal save <Name> <?> → Tor-Name (Freitext)
|
|
||||||
case "save" -> suggestions.add("<TorName>");
|
case "save" -> suggestions.add("<TorName>");
|
||||||
// /nexuslobby ball goal delete <Name>
|
|
||||||
case "delete" -> {
|
case "delete" -> {
|
||||||
// Tor-Namen aus Config laden und vorschlagen
|
|
||||||
var section = NexusLobby.getInstance().getConfig()
|
var section = NexusLobby.getInstance().getConfig()
|
||||||
.getConfigurationSection("ball.goals");
|
.getConfigurationSection("ball.goals");
|
||||||
if (section != null) suggestions.addAll(section.getKeys(false));
|
if (section != null) suggestions.addAll(section.getKeys(false));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// /nexuslobby ball goal save <Name> <?> → Team 1 oder 2
|
// /nexuslobby ball goal save <Name> <?> → Team 1 oder 2
|
||||||
} else if (args.length == 5) {
|
} else if (args.length == 5) {
|
||||||
if (args[0].equalsIgnoreCase("ball")
|
if (args[0].equalsIgnoreCase("ball")
|
||||||
&& args[1].equalsIgnoreCase("goal")
|
&& args[1].equalsIgnoreCase("goal")
|
||||||
@@ -111,15 +120,13 @@ public class LobbyTabCompleter implements TabCompleter {
|
|||||||
if (args.length == 1) {
|
if (args.length == 1) {
|
||||||
suggestions.addAll(Arrays.asList("create", "delete"));
|
suggestions.addAll(Arrays.asList("create", "delete"));
|
||||||
} else if (args.length == 2 && args[0].equalsIgnoreCase("delete")) {
|
} else if (args.length == 2 && args[0].equalsIgnoreCase("delete")) {
|
||||||
if (hologramModule != null)
|
if (hologramModule != null) suggestions.addAll(hologramModule.getHologramIds());
|
||||||
suggestions.addAll(hologramModule.getHologramIds());
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ── Wartungsmodus ─────────────────────────────────────────────────────
|
// ── Wartungsmodus ─────────────────────────────────────────────────────
|
||||||
else if (cmdName.equals("maintenance")) {
|
else if (cmdName.equals("maintenance")) {
|
||||||
if (args.length == 1)
|
if (args.length == 1) suggestions.addAll(Arrays.asList("on", "off"));
|
||||||
suggestions.addAll(Arrays.asList("on", "off"));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ── Portalsystem ──────────────────────────────────────────────────────
|
// ── Portalsystem ──────────────────────────────────────────────────────
|
||||||
@@ -127,8 +134,7 @@ public class LobbyTabCompleter implements TabCompleter {
|
|||||||
if (args.length == 1) {
|
if (args.length == 1) {
|
||||||
suggestions.addAll(Arrays.asList("create", "delete", "list"));
|
suggestions.addAll(Arrays.asList("create", "delete", "list"));
|
||||||
} else if (args.length == 2 && args[0].equalsIgnoreCase("delete")) {
|
} else if (args.length == 2 && args[0].equalsIgnoreCase("delete")) {
|
||||||
if (portalManager != null)
|
if (portalManager != null) suggestions.addAll(portalManager.getPortalNames());
|
||||||
suggestions.addAll(portalManager.getPortalNames());
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -143,8 +149,7 @@ public class LobbyTabCompleter implements TabCompleter {
|
|||||||
|
|
||||||
// ── Intro System ──────────────────────────────────────────────────────
|
// ── Intro System ──────────────────────────────────────────────────────
|
||||||
else if (cmdName.equals("intro")) {
|
else if (cmdName.equals("intro")) {
|
||||||
if (args.length == 1)
|
if (args.length == 1) suggestions.addAll(Arrays.asList("add", "clear", "start"));
|
||||||
suggestions.addAll(Arrays.asList("add", "clear", "start"));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ── WorldBorder ───────────────────────────────────────────────────────
|
// ── WorldBorder ───────────────────────────────────────────────────────
|
||||||
@@ -160,7 +165,7 @@ public class LobbyTabCompleter implements TabCompleter {
|
|||||||
else if (cmdName.equals("nexuscmd") || cmdName.equals("ncmd")
|
else if (cmdName.equals("nexuscmd") || cmdName.equals("ncmd")
|
||||||
|| cmdName.equals("ascmd") || cmdName.equals("conv")) {
|
|| cmdName.equals("ascmd") || cmdName.equals("conv")) {
|
||||||
if (args.length == 1) {
|
if (args.length == 1) {
|
||||||
suggestions.addAll(Arrays.asList("add", "remove", "list", "name", "lookat", "conv", "say"));
|
suggestions.addAll(Arrays.asList("add", "remove", "list", "name", "lookat", "conv", "say", "clearbubbles"));
|
||||||
} else if (args.length == 2) {
|
} else if (args.length == 2) {
|
||||||
switch (args[0].toLowerCase()) {
|
switch (args[0].toLowerCase()) {
|
||||||
case "add" -> suggestions.addAll(Arrays.asList("0","1","2","3","4","5","6","7","8","9"));
|
case "add" -> suggestions.addAll(Arrays.asList("0","1","2","3","4","5","6","7","8","9"));
|
||||||
@@ -201,7 +206,6 @@ public class LobbyTabCompleter implements TabCompleter {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Filtert die Liste basierend auf der bisherigen Eingabe
|
|
||||||
return suggestions.stream()
|
return suggestions.stream()
|
||||||
.filter(s -> s.toLowerCase().startsWith(args[args.length - 1].toLowerCase()))
|
.filter(s -> s.toLowerCase().startsWith(args[args.length - 1].toLowerCase()))
|
||||||
.collect(Collectors.toList());
|
.collect(Collectors.toList());
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ package de.nexuslobby.commands;
|
|||||||
|
|
||||||
import de.nexuslobby.NexusLobby;
|
import de.nexuslobby.NexusLobby;
|
||||||
import de.nexuslobby.modules.ScoreboardModule;
|
import de.nexuslobby.modules.ScoreboardModule;
|
||||||
|
import de.nexuslobby.modules.hologram.HologramModule;
|
||||||
import de.nexuslobby.modules.parkour.ParkourManager;
|
import de.nexuslobby.modules.parkour.ParkourManager;
|
||||||
import org.bukkit.Bukkit;
|
import org.bukkit.Bukkit;
|
||||||
import org.bukkit.Location;
|
import org.bukkit.Location;
|
||||||
@@ -22,7 +23,7 @@ public class NexusLobbyCommand implements CommandExecutor {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean onCommand(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) {
|
public boolean onCommand(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) {
|
||||||
|
|
||||||
if (!(sender instanceof Player player)) {
|
if (!(sender instanceof Player player)) {
|
||||||
sender.sendMessage(de.nexuslobby.utils.LangManager.get("only_player"));
|
sender.sendMessage(de.nexuslobby.utils.LangManager.get("only_player"));
|
||||||
return true;
|
return true;
|
||||||
@@ -34,19 +35,18 @@ public class NexusLobbyCommand implements CommandExecutor {
|
|||||||
// --- DIREKTE KURZ-BEFEHLE ---
|
// --- DIREKTE KURZ-BEFEHLE ---
|
||||||
if (cmdName.equalsIgnoreCase("setstart")) {
|
if (cmdName.equalsIgnoreCase("setstart")) {
|
||||||
if (!player.hasPermission("nexuslobby.admin")) return noPerm(player);
|
if (!player.hasPermission("nexuslobby.admin")) return noPerm(player);
|
||||||
handleSetStart(player, pm);
|
// Standard: Strecke 1 setzen
|
||||||
|
pm.setStartLocation(player, 1);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (cmdName.equalsIgnoreCase("setcheckpoint")) {
|
if (cmdName.equalsIgnoreCase("setcheckpoint")) {
|
||||||
if (!player.hasPermission("nexuslobby.admin")) return noPerm(player);
|
if (!player.hasPermission("nexuslobby.admin")) return noPerm(player);
|
||||||
pm.setCheckpoint(player, player.getLocation());
|
pm.setCheckpoint(player, 1);
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("parkour_checkpoint_set"));
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (cmdName.equalsIgnoreCase("setfinish")) {
|
if (cmdName.equalsIgnoreCase("setfinish")) {
|
||||||
if (!player.hasPermission("nexuslobby.admin")) return noPerm(player);
|
if (!player.hasPermission("nexuslobby.admin")) return noPerm(player);
|
||||||
pm.setFinishLocation(player.getLocation());
|
pm.setFinishLocation(player, 1);
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("parkour_finish_set"));
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -75,6 +75,7 @@ public class NexusLobbyCommand implements CommandExecutor {
|
|||||||
}
|
}
|
||||||
|
|
||||||
switch (args[0].toLowerCase()) {
|
switch (args[0].toLowerCase()) {
|
||||||
|
|
||||||
case "reload":
|
case "reload":
|
||||||
if (!player.hasPermission("nexuslobby.admin")) return noPerm(player);
|
if (!player.hasPermission("nexuslobby.admin")) return noPerm(player);
|
||||||
NexusLobby.getInstance().reloadPlugin();
|
NexusLobby.getInstance().reloadPlugin();
|
||||||
@@ -83,7 +84,6 @@ public class NexusLobbyCommand implements CommandExecutor {
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case "cleanbubbles":
|
case "cleanbubbles":
|
||||||
// Bereinigt alle hängengebliebenen Sprechblasen-ArmorStands im gesamten Server.
|
|
||||||
if (!player.hasPermission("nexuslobby.admin")) return noPerm(player);
|
if (!player.hasPermission("nexuslobby.admin")) return noPerm(player);
|
||||||
handleCleanBubbles(player);
|
handleCleanBubbles(player);
|
||||||
break;
|
break;
|
||||||
@@ -96,7 +96,7 @@ public class NexusLobbyCommand implements CommandExecutor {
|
|||||||
config.set("spawn.x", loc.getX());
|
config.set("spawn.x", loc.getX());
|
||||||
config.set("spawn.y", loc.getY());
|
config.set("spawn.y", loc.getY());
|
||||||
config.set("spawn.z", loc.getZ());
|
config.set("spawn.z", loc.getZ());
|
||||||
config.set("spawn.yaw", (double) loc.getYaw());
|
config.set("spawn.yaw", (double) loc.getYaw());
|
||||||
config.set("spawn.pitch", (double) loc.getPitch());
|
config.set("spawn.pitch", (double) loc.getPitch());
|
||||||
NexusLobby.getInstance().saveConfig();
|
NexusLobby.getInstance().saveConfig();
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("spawn_set"));
|
player.sendMessage(de.nexuslobby.utils.LangManager.get("spawn_set"));
|
||||||
@@ -117,7 +117,12 @@ public class NexusLobbyCommand implements CommandExecutor {
|
|||||||
handleScoreboard(player, args);
|
handleScoreboard(player, args);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case "ball": // NEU: Weiterleitung an das SoccerModule
|
case "gadgetshield":
|
||||||
|
if (!player.hasPermission("nexuslobby.admin") && !player.isOp()) return noPerm(player);
|
||||||
|
de.nexuslobby.modules.gadgets.GadgetShield.toggle(player);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case "ball":
|
||||||
if (NexusLobby.getInstance().getSoccerModule() != null) {
|
if (NexusLobby.getInstance().getSoccerModule() != null) {
|
||||||
return NexusLobby.getInstance().getSoccerModule().onCommand(sender, command, label, args);
|
return NexusLobby.getInstance().getSoccerModule().onCommand(sender, command, label, args);
|
||||||
}
|
}
|
||||||
@@ -125,42 +130,7 @@ public class NexusLobbyCommand implements CommandExecutor {
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case "parkour":
|
case "parkour":
|
||||||
if (args.length < 2) {
|
handleParkour(player, args, pm);
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("parkour_usage"));
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
String sub = args[1].toLowerCase();
|
|
||||||
if (!player.hasPermission("nexuslobby.admin") && !sub.equals("reset")) return noPerm(player);
|
|
||||||
|
|
||||||
switch (sub) {
|
|
||||||
case "setstart":
|
|
||||||
handleSetStart(player, pm);
|
|
||||||
break;
|
|
||||||
case "setfinish":
|
|
||||||
pm.setFinishLocation(player.getLocation());
|
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("parkour_finish_set"));
|
|
||||||
break;
|
|
||||||
case "setcheckpoint":
|
|
||||||
pm.setCheckpoint(player, player.getLocation());
|
|
||||||
break;
|
|
||||||
case "reset":
|
|
||||||
pm.stopParkour(player);
|
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("parkour_run_aborted"));
|
|
||||||
break;
|
|
||||||
case "clear":
|
|
||||||
pm.clearStats();
|
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("parkour_besttimes_cleared"));
|
|
||||||
break;
|
|
||||||
case "removeall":
|
|
||||||
pm.removeAllPoints();
|
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("parkour_track_removed"));
|
|
||||||
player.playSound(player.getLocation(), Sound.ENTITY_ITEM_BREAK, 1f, 1f);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("unknown_subcommand"));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
@@ -171,13 +141,128 @@ public class NexusLobbyCommand implements CommandExecutor {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void handleSetStart(Player player, ParkourManager pm) {
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
// NPC Erkennung (Blickrichtung auf ArmorStand)
|
// Parkour-Handler
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
private void handleParkour(Player player, String[] args, ParkourManager pm) {
|
||||||
|
if (args.length < 2) {
|
||||||
|
player.sendMessage(de.nexuslobby.utils.LangManager.get("parkour_usage"));
|
||||||
|
// Strecken-Info ausgeben
|
||||||
|
player.sendMessage(pm.getTrackInfo());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
String sub = args[1].toLowerCase();
|
||||||
|
|
||||||
|
// Nur "reset" kann auch ohne Admin-Recht ausgeführt werden (eigenen Lauf abbrechen)
|
||||||
|
if (!player.hasPermission("nexuslobby.admin") && !sub.equals("reset")) {
|
||||||
|
noPerm(player);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (sub) {
|
||||||
|
|
||||||
|
// /nexus parkour setstart <1|2>
|
||||||
|
case "setstart": {
|
||||||
|
int track = parseTrack(args, 2, player);
|
||||||
|
if (track == -1) return;
|
||||||
|
handleSetStart(player, pm, track);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// /nexus parkour setfinish <1|2>
|
||||||
|
case "setfinish": {
|
||||||
|
int track = parseTrack(args, 2, player);
|
||||||
|
if (track == -1) return;
|
||||||
|
pm.setFinishLocation(player, track);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// /nexus parkour setcheckpoint <1|2>
|
||||||
|
case "setcheckpoint": {
|
||||||
|
int track = parseTrack(args, 2, player);
|
||||||
|
if (track == -1) return;
|
||||||
|
pm.setCheckpoint(player, track);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case "reset":
|
||||||
|
pm.stopParkour(player);
|
||||||
|
player.sendMessage(de.nexuslobby.utils.LangManager.get("parkour_run_aborted"));
|
||||||
|
break;
|
||||||
|
|
||||||
|
case "clear":
|
||||||
|
pm.clearStats();
|
||||||
|
player.sendMessage(de.nexuslobby.utils.LangManager.get("parkour_besttimes_cleared"));
|
||||||
|
break;
|
||||||
|
|
||||||
|
case "removeall":
|
||||||
|
pm.removeAllPoints();
|
||||||
|
player.sendMessage(de.nexuslobby.utils.LangManager.get("parkour_track_removed"));
|
||||||
|
player.playSound(player.getLocation(), Sound.ENTITY_ITEM_BREAK, 1f, 1f);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case "info":
|
||||||
|
player.sendMessage(pm.getTrackInfo());
|
||||||
|
break;
|
||||||
|
|
||||||
|
// /nexus parkour holo <1|2> – Leaderboard-Hologramm an aktueller Position setzen
|
||||||
|
case "holo": {
|
||||||
|
int track = parseTrack(args, 2, player);
|
||||||
|
if (track == -1) return;
|
||||||
|
HologramModule hm = NexusLobby.getInstance().getHologramModule();
|
||||||
|
if (hm == null) {
|
||||||
|
player.sendMessage("§cHologram-Modul nicht verfügbar!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
String holoId = "parkour_top_" + track;
|
||||||
|
List<String> pages = new java.util.ArrayList<>();
|
||||||
|
pages.add("%nexuslobby_parkour_top_" + track + "%");
|
||||||
|
hm.createHologram(holoId, player.getLocation(), pages);
|
||||||
|
player.sendMessage("§8[§6Parkour§8] §aLeaderboard-Hologramm für §eStrecke " + track
|
||||||
|
+ " §agesetzt! §8(ID: §7" + holoId + "§8)");
|
||||||
|
player.sendMessage("§8[§6Parkour§8] §7Das Hologramm aktualisiert sich automatisch via PlaceholderAPI.");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
player.sendMessage(de.nexuslobby.utils.LangManager.get("unknown_subcommand"));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Liest die Track-Nummer aus args[index].
|
||||||
|
* Gibt 1 oder 2 zurück; bei Fehler sendet er eine Nachricht und gibt -1 zurück.
|
||||||
|
*/
|
||||||
|
private int parseTrack(String[] args, int index, Player player) {
|
||||||
|
if (args.length <= index) {
|
||||||
|
player.sendMessage("§8[§6Nexus§8] §cBitte gib eine Strecken-Nummer an: §e1 §7oder §e2");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
int track;
|
||||||
|
try {
|
||||||
|
track = Integer.parseInt(args[index]);
|
||||||
|
} catch (NumberFormatException e) {
|
||||||
|
player.sendMessage("§8[§6Nexus§8] §cUngültige Strecken-Nummer. Bitte §e1 §7oder §e2 §7verwenden.");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (track < 1 || track > 2) {
|
||||||
|
player.sendMessage("§8[§6Nexus§8] §cNur Strecke §e1 §7oder §e2 §7sind erlaubt.");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return track;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void handleSetStart(Player player, ParkourManager pm, int track) {
|
||||||
|
// NPC-Erkennung: schaut der Spieler auf einen ArmorStand?
|
||||||
ArmorStand targetAs = null;
|
ArmorStand targetAs = null;
|
||||||
List<Entity> nearby = player.getNearbyEntities(4, 4, 4);
|
List<Entity> nearby = player.getNearbyEntities(4, 4, 4);
|
||||||
for (Entity e : nearby) {
|
for (Entity e : nearby) {
|
||||||
if (e instanceof ArmorStand as) {
|
if (e instanceof ArmorStand as) {
|
||||||
double dot = player.getLocation().getDirection().dot(as.getLocation().toVector().subtract(player.getLocation().toVector()).normalize());
|
double dot = player.getLocation().getDirection()
|
||||||
|
.dot(as.getLocation().toVector().subtract(player.getLocation().toVector()).normalize());
|
||||||
if (dot > 0.9) {
|
if (dot > 0.9) {
|
||||||
targetAs = as;
|
targetAs = as;
|
||||||
break;
|
break;
|
||||||
@@ -189,13 +274,16 @@ public class NexusLobbyCommand implements CommandExecutor {
|
|||||||
targetAs.addScoreboardTag("parkour_npc");
|
targetAs.addScoreboardTag("parkour_npc");
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("parkour_npc_marked"));
|
player.sendMessage(de.nexuslobby.utils.LangManager.get("parkour_npc_marked"));
|
||||||
}
|
}
|
||||||
pm.setStartLocation(player.getLocation());
|
pm.setStartLocation(player, track);
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("parkour_start_set"));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
// Sonstige Handler
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
private void handleCleanBubbles(Player player) {
|
private void handleCleanBubbles(Player player) {
|
||||||
de.nexuslobby.modules.armorstandtools.ConversationManager cm =
|
de.nexuslobby.modules.armorstandtools.ConversationManager cm =
|
||||||
NexusLobby.getInstance().getConversationManager();
|
NexusLobby.getInstance().getConversationManager();
|
||||||
if (cm == null) {
|
if (cm == null) {
|
||||||
player.sendMessage("§8[§6Nexus§8] §cConversationManager ist nicht aktiv.");
|
player.sendMessage("§8[§6Nexus§8] §cConversationManager ist nicht aktiv.");
|
||||||
return;
|
return;
|
||||||
@@ -220,18 +308,17 @@ public class NexusLobbyCommand implements CommandExecutor {
|
|||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("scoreboard_module_disabled"));
|
player.sendMessage(de.nexuslobby.utils.LangManager.get("scoreboard_module_disabled"));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
String sub = args[1].toLowerCase();
|
switch (args[1].toLowerCase()) {
|
||||||
switch (sub) {
|
case "on" -> sbModule.setVisibility(player, true);
|
||||||
case "on": sbModule.setVisibility(player, true); break;
|
case "off" -> sbModule.setVisibility(player, false);
|
||||||
case "off": sbModule.setVisibility(player, false); break;
|
case "admin" -> {
|
||||||
case "admin":
|
|
||||||
if (player.hasPermission("nexuslobby.scoreboard.admin")) sbModule.setAdminMode(player, true);
|
if (player.hasPermission("nexuslobby.scoreboard.admin")) sbModule.setAdminMode(player, true);
|
||||||
else player.sendMessage(de.nexuslobby.utils.LangManager.get("no_permission"));
|
else player.sendMessage(de.nexuslobby.utils.LangManager.get("no_permission"));
|
||||||
break;
|
}
|
||||||
case "spieler":
|
case "spieler" -> {
|
||||||
if (player.hasPermission("nexuslobby.scoreboard.admin")) sbModule.setAdminMode(player, false);
|
if (player.hasPermission("nexuslobby.scoreboard.admin")) sbModule.setAdminMode(player, false);
|
||||||
else player.sendMessage(de.nexuslobby.utils.LangManager.get("no_permission"));
|
else player.sendMessage(de.nexuslobby.utils.LangManager.get("no_permission"));
|
||||||
break;
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -248,13 +335,20 @@ public class NexusLobbyCommand implements CommandExecutor {
|
|||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("info_title"));
|
player.sendMessage(de.nexuslobby.utils.LangManager.get("info_title"));
|
||||||
player.sendMessage("");
|
player.sendMessage("");
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("info_spawn"));
|
player.sendMessage(de.nexuslobby.utils.LangManager.get("info_spawn"));
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("info_parkour"));
|
player.sendMessage("§e/nexus parkour setstart <1|2> §7- Start setzen");
|
||||||
|
player.sendMessage("§e/nexus parkour setcheckpoint <1|2> §7- Checkpoint setzen");
|
||||||
|
player.sendMessage("§e/nexus parkour setfinish <1|2> §7- Ziel setzen");
|
||||||
|
player.sendMessage("§e/nexus parkour holo <1|2> §7- Top-10 Hologramm setzen");
|
||||||
|
player.sendMessage("§e/nexus parkour reset §7- Eigenen Run abbrechen");
|
||||||
|
player.sendMessage("§e/nexus parkour clear §7- §cTop-10 Liste löschen");
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("info_removeall"));
|
player.sendMessage(de.nexuslobby.utils.LangManager.get("info_removeall"));
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("info_ball"));
|
player.sendMessage(de.nexuslobby.utils.LangManager.get("info_ball"));
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("info_setspawn"));
|
player.sendMessage(de.nexuslobby.utils.LangManager.get("info_setspawn"));
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("info_scoreboard"));
|
player.sendMessage(de.nexuslobby.utils.LangManager.get("info_scoreboard"));
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("info_reload"));
|
player.sendMessage(de.nexuslobby.utils.LangManager.get("info_reload"));
|
||||||
player.sendMessage("§e/nexuslobby cleanbubbles §7- Hängende Sprechblasen entfernen");
|
player.sendMessage("§e/nexuslobby cleanbubbles §7- Hängende Sprechblasen entfernen");
|
||||||
|
if (player.hasPermission("nexuslobby.admin") || player.isOp())
|
||||||
|
player.sendMessage("§e/nexuslobby gadgetshield §7- Gadget-Schutz für Admins ein/ausschalten");
|
||||||
player.sendMessage(de.nexuslobby.utils.LangManager.get("info_footer"));
|
player.sendMessage(de.nexuslobby.utils.LangManager.get("info_footer"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -2,6 +2,7 @@ package de.nexuslobby.modules;
|
|||||||
|
|
||||||
import de.nexuslobby.NexusLobby;
|
import de.nexuslobby.NexusLobby;
|
||||||
import de.nexuslobby.api.Module;
|
import de.nexuslobby.api.Module;
|
||||||
|
import io.papermc.paper.scoreboard.numbers.NumberFormat;
|
||||||
import me.clip.placeholderapi.PlaceholderAPI;
|
import me.clip.placeholderapi.PlaceholderAPI;
|
||||||
import org.bukkit.Bukkit;
|
import org.bukkit.Bukkit;
|
||||||
import org.bukkit.ChatColor;
|
import org.bukkit.ChatColor;
|
||||||
@@ -9,9 +10,12 @@ import org.bukkit.configuration.file.FileConfiguration;
|
|||||||
import org.bukkit.entity.Player;
|
import org.bukkit.entity.Player;
|
||||||
import org.bukkit.scheduler.BukkitRunnable;
|
import org.bukkit.scheduler.BukkitRunnable;
|
||||||
import org.bukkit.scoreboard.*;
|
import org.bukkit.scoreboard.*;
|
||||||
|
import org.bukkit.scoreboard.Criteria;
|
||||||
|
|
||||||
|
import java.util.HashMap;
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
|
||||||
@@ -23,10 +27,12 @@ public class ScoreboardModule implements Module, Listener {
|
|||||||
|
|
||||||
private final NexusLobby plugin = NexusLobby.getInstance();
|
private final NexusLobby plugin = NexusLobby.getInstance();
|
||||||
private boolean placeholderAPIEnabled;
|
private boolean placeholderAPIEnabled;
|
||||||
|
|
||||||
// Speicher für die aktuellen Spieler-Einstellungen (bis zum Restart/Reload)
|
|
||||||
private final Set<UUID> hiddenPlayers = new HashSet<>();
|
private final Set<UUID> hiddenPlayers = new HashSet<>();
|
||||||
private final Set<UUID> adminModePlayers = new HashSet<>();
|
private final Set<UUID> adminModePlayers = new HashSet<>();
|
||||||
|
|
||||||
|
// WICHTIG: Hier speichern wir die Scoreboards pro Spieler, um sie wiederzuverwenden
|
||||||
|
private final Map<UUID, Scoreboard> playerBoards = new HashMap<>();
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String getName() {
|
public String getName() {
|
||||||
@@ -38,7 +44,6 @@ public class ScoreboardModule implements Module, Listener {
|
|||||||
FileConfiguration vConfig = plugin.getVisualsConfig();
|
FileConfiguration vConfig = plugin.getVisualsConfig();
|
||||||
if (!vConfig.getBoolean("scoreboard.enabled", true)) return;
|
if (!vConfig.getBoolean("scoreboard.enabled", true)) return;
|
||||||
|
|
||||||
// Listener registrieren
|
|
||||||
Bukkit.getPluginManager().registerEvents(this, plugin);
|
Bukkit.getPluginManager().registerEvents(this, plugin);
|
||||||
|
|
||||||
placeholderAPIEnabled = Bukkit.getPluginManager().getPlugin("PlaceholderAPI") != null;
|
placeholderAPIEnabled = Bukkit.getPluginManager().getPlugin("PlaceholderAPI") != null;
|
||||||
@@ -53,9 +58,6 @@ public class ScoreboardModule implements Module, Listener {
|
|||||||
}.runTaskTimer(plugin, 0L, vConfig.getLong("scoreboard.update_ticks", 20L));
|
}.runTaskTimer(plugin, 0L, vConfig.getLong("scoreboard.update_ticks", 20L));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Setzt Scoreboard-Status beim Join gemäß config.yml (scoreboard-default-visible)
|
|
||||||
*/
|
|
||||||
@EventHandler
|
@EventHandler
|
||||||
public void onPlayerJoin(PlayerJoinEvent event) {
|
public void onPlayerJoin(PlayerJoinEvent event) {
|
||||||
Player player = event.getPlayer();
|
Player player = event.getPlayer();
|
||||||
@@ -63,13 +65,19 @@ public class ScoreboardModule implements Module, Listener {
|
|||||||
if (!defaultVisible) {
|
if (!defaultVisible) {
|
||||||
hiddenPlayers.add(player.getUniqueId());
|
hiddenPlayers.add(player.getUniqueId());
|
||||||
player.setScoreboard(Bukkit.getScoreboardManager().getMainScoreboard());
|
player.setScoreboard(Bukkit.getScoreboardManager().getMainScoreboard());
|
||||||
|
// Sicherstellen, dass kein altes Board gespeichert ist
|
||||||
|
playerBoards.remove(player.getUniqueId());
|
||||||
} else {
|
} else {
|
||||||
hiddenPlayers.remove(player.getUniqueId());
|
hiddenPlayers.remove(player.getUniqueId());
|
||||||
|
// Beim Einloggen initialisieren, damit es nicht beim ersten Tick flackert
|
||||||
|
if (!playerBoards.containsKey(player.getUniqueId())) {
|
||||||
|
playerBoards.put(player.getUniqueId(), Bukkit.getScoreboardManager().getNewScoreboard());
|
||||||
|
player.setScoreboard(playerBoards.get(player.getUniqueId()));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void updateSidebar(Player player) {
|
private void updateSidebar(Player player) {
|
||||||
// 1. Prüfen, ob der Spieler das Scoreboard ausgeblendet hat
|
|
||||||
if (hiddenPlayers.contains(player.getUniqueId())) {
|
if (hiddenPlayers.contains(player.getUniqueId())) {
|
||||||
if (player.getScoreboard() != Bukkit.getScoreboardManager().getMainScoreboard()) {
|
if (player.getScoreboard() != Bukkit.getScoreboardManager().getMainScoreboard()) {
|
||||||
player.setScoreboard(Bukkit.getScoreboardManager().getMainScoreboard());
|
player.setScoreboard(Bukkit.getScoreboardManager().getMainScoreboard());
|
||||||
@@ -78,49 +86,81 @@ public class ScoreboardModule implements Module, Listener {
|
|||||||
}
|
}
|
||||||
|
|
||||||
FileConfiguration vConfig = plugin.getVisualsConfig();
|
FileConfiguration vConfig = plugin.getVisualsConfig();
|
||||||
Scoreboard board = player.getScoreboard();
|
|
||||||
|
String configPath = (adminModePlayers.contains(player.getUniqueId())
|
||||||
|
&& player.hasPermission("nexuslobby.scoreboard.admin"))
|
||||||
|
? "scoreboard.owner"
|
||||||
|
: "scoreboard.default";
|
||||||
|
|
||||||
|
String title = vConfig.getString(configPath + ".title", "&6&lNexusLobby");
|
||||||
|
List<String> lines = vConfig.getStringList(configPath + ".lines");
|
||||||
|
|
||||||
|
// ---------------------------------------------------------
|
||||||
|
// FIX: Statt getNewScoreboard() holen wir das gespeicherte Board
|
||||||
|
// oder erstellen es genau einmal beim ersten Mal.
|
||||||
|
// ---------------------------------------------------------
|
||||||
|
Scoreboard board = playerBoards.get(player.getUniqueId());
|
||||||
|
|
||||||
if (board == Bukkit.getScoreboardManager().getMainScoreboard()) {
|
if (board == null) {
|
||||||
board = Bukkit.getScoreboardManager().getNewScoreboard();
|
board = Bukkit.getScoreboardManager().getNewScoreboard();
|
||||||
|
playerBoards.put(player.getUniqueId(), board);
|
||||||
player.setScoreboard(board);
|
player.setScoreboard(board);
|
||||||
}
|
}
|
||||||
|
|
||||||
Objective obj = board.getObjective("lobby");
|
// Altes Objective entfernen
|
||||||
if (obj == null) {
|
Objective oldObj = board.getObjective("lobby");
|
||||||
obj = board.registerNewObjective("lobby", "dummy", "title");
|
if (oldObj != null) oldObj.unregister();
|
||||||
obj.setDisplaySlot(DisplaySlot.SIDEBAR);
|
|
||||||
|
// Alle alten Zeilen-Teams sicher entfernen (großzügige Obergrenze)
|
||||||
|
for (int i = 0; i < Math.max(lines.size() + 5, 32); i++) {
|
||||||
|
Team old = board.getTeam("line_" + i);
|
||||||
|
if (old != null) old.unregister();
|
||||||
}
|
}
|
||||||
|
|
||||||
// 2. Pfad-Logik basierend auf dem Modus (/nexus sb admin/spieler)
|
// Paper 1.21: Objective mit Component-Titel registrieren
|
||||||
String configPath = "scoreboard.default";
|
String translatedTitle = translate(player, title);
|
||||||
|
Objective obj = board.registerNewObjective(
|
||||||
// Wenn im Admin-Modus UND Rechte vorhanden -> owner Sektion
|
"lobby",
|
||||||
if (adminModePlayers.contains(player.getUniqueId()) && player.hasPermission("nexuslobby.scoreboard.admin")) {
|
Criteria.DUMMY,
|
||||||
configPath = "scoreboard.owner";
|
net.kyori.adventure.text.serializer.legacy.LegacyComponentSerializer
|
||||||
} else {
|
.legacySection()
|
||||||
// Standardmäßig default nutzen
|
.deserialize(translatedTitle)
|
||||||
configPath = "scoreboard.default";
|
);
|
||||||
}
|
obj.setDisplaySlot(DisplaySlot.SIDEBAR);
|
||||||
|
|
||||||
String title = vConfig.getString(configPath + ".title", "&6&lNexusLobby");
|
|
||||||
obj.setDisplayName(translate(player, title));
|
|
||||||
|
|
||||||
List<String> lines = vConfig.getStringList(configPath + ".lines");
|
// Zahlen global für dieses Objective ausblenden (Paper 1.20.4+ API)
|
||||||
|
obj.numberFormat(NumberFormat.blank());
|
||||||
// Scores zurücksetzen um Duplikate zu vermeiden
|
|
||||||
board.getEntries().forEach(board::resetScores);
|
|
||||||
|
|
||||||
for (int i = 0; i < lines.size(); i++) {
|
for (int i = 0; i < lines.size(); i++) {
|
||||||
String line = translate(player, lines.get(i));
|
String lineText = translate(player, lines.get(i));
|
||||||
|
|
||||||
// Verhindert das Verschwinden leerer Zeilen
|
// Eindeutiger unsichtbarer Entry pro Zeile via Farb-Code
|
||||||
if (line.isEmpty() || line.trim().isEmpty()) {
|
String entry = ChatColor.values()[i % ChatColor.values().length].toString()
|
||||||
line = "" + ChatColor.values()[i] + ChatColor.RESET;
|
+ ChatColor.RESET;
|
||||||
|
|
||||||
|
// getTeam statt registerNewTeam — verhindert "already in use" Fehler
|
||||||
|
Team team = board.getTeam("line_" + i);
|
||||||
|
if (team == null) team = board.registerNewTeam("line_" + i);
|
||||||
|
team.getEntries().forEach(team::removeEntry);
|
||||||
|
team.addEntry(entry);
|
||||||
|
|
||||||
|
// Text aufteilen falls länger als 64 Zeichen
|
||||||
|
if (lineText.length() > 64) {
|
||||||
|
team.setPrefix(lineText.substring(0, 64));
|
||||||
|
team.setSuffix(lineText.substring(64, Math.min(lineText.length(), 128)));
|
||||||
|
} else {
|
||||||
|
team.setPrefix(lineText);
|
||||||
|
team.setSuffix("");
|
||||||
}
|
}
|
||||||
|
|
||||||
Score score = obj.getScore(line);
|
// Score setzen + Zahl pro Eintrag zusätzlich ausblenden
|
||||||
|
org.bukkit.scoreboard.Score score = obj.getScore(entry);
|
||||||
score.setScore(lines.size() - i);
|
score.setScore(lines.size() - i);
|
||||||
|
score.numberFormat(NumberFormat.blank());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Kein player.setScoreboard(board) nötig, da wir das Objekt 'board'
|
||||||
|
// direkt verändert haben, das der Spieler bereits hat.
|
||||||
}
|
}
|
||||||
|
|
||||||
// --- Methoden für den /nexus sb Befehl ---
|
// --- Methoden für den /nexus sb Befehl ---
|
||||||
@@ -128,7 +168,13 @@ public class ScoreboardModule implements Module, Listener {
|
|||||||
public void setVisibility(Player player, boolean visible) {
|
public void setVisibility(Player player, boolean visible) {
|
||||||
if (visible) {
|
if (visible) {
|
||||||
hiddenPlayers.remove(player.getUniqueId());
|
hiddenPlayers.remove(player.getUniqueId());
|
||||||
|
// Wenn wir es wieder einschalten, ggf. Board zurücksetzen oder neu holen
|
||||||
|
if (!playerBoards.containsKey(player.getUniqueId())) {
|
||||||
|
playerBoards.put(player.getUniqueId(), Bukkit.getScoreboardManager().getNewScoreboard());
|
||||||
|
}
|
||||||
|
player.setScoreboard(playerBoards.get(player.getUniqueId()));
|
||||||
player.sendMessage("§7[§6Nexus§7] §aScoreboard eingeschaltet.");
|
player.sendMessage("§7[§6Nexus§7] §aScoreboard eingeschaltet.");
|
||||||
|
updateSidebar(player); // Sofort aktualisieren
|
||||||
} else {
|
} else {
|
||||||
hiddenPlayers.add(player.getUniqueId());
|
hiddenPlayers.add(player.getUniqueId());
|
||||||
player.setScoreboard(Bukkit.getScoreboardManager().getMainScoreboard());
|
player.setScoreboard(Bukkit.getScoreboardManager().getMainScoreboard());
|
||||||
@@ -144,7 +190,6 @@ public class ScoreboardModule implements Module, Listener {
|
|||||||
adminModePlayers.remove(player.getUniqueId());
|
adminModePlayers.remove(player.getUniqueId());
|
||||||
player.sendMessage("§7[§6Nexus§7] §eModus: §6Spieler-Scoreboard §7(Default-Sektion)");
|
player.sendMessage("§7[§6Nexus§7] §eModus: §6Spieler-Scoreboard §7(Default-Sektion)");
|
||||||
}
|
}
|
||||||
// Sofortiges Update erzwingen
|
|
||||||
updateSidebar(player);
|
updateSidebar(player);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -153,9 +198,7 @@ public class ScoreboardModule implements Module, Listener {
|
|||||||
if (placeholderAPIEnabled) {
|
if (placeholderAPIEnabled) {
|
||||||
try {
|
try {
|
||||||
translated = PlaceholderAPI.setPlaceholders(player, text);
|
translated = PlaceholderAPI.setPlaceholders(player, text);
|
||||||
} catch (NoClassDefFoundError ignored) {
|
} catch (NoClassDefFoundError ignored) {}
|
||||||
// PlaceholderAPI fehlt zur Laufzeit
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return ChatColor.translateAlternateColorCodes('&', translated);
|
return ChatColor.translateAlternateColorCodes('&', translated);
|
||||||
}
|
}
|
||||||
@@ -168,5 +211,6 @@ public class ScoreboardModule implements Module, Listener {
|
|||||||
}
|
}
|
||||||
hiddenPlayers.clear();
|
hiddenPlayers.clear();
|
||||||
adminModePlayers.clear();
|
adminModePlayers.clear();
|
||||||
|
playerBoards.clear(); // Speicher freigeben
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -6,11 +6,15 @@ import de.nexuslobby.NexusLobby;
|
|||||||
import org.bukkit.Bukkit;
|
import org.bukkit.Bukkit;
|
||||||
import org.bukkit.Material;
|
import org.bukkit.Material;
|
||||||
import org.bukkit.entity.ArmorStand;
|
import org.bukkit.entity.ArmorStand;
|
||||||
|
import org.bukkit.entity.Entity;
|
||||||
import org.bukkit.entity.Player;
|
import org.bukkit.entity.Player;
|
||||||
|
import org.bukkit.entity.Projectile;
|
||||||
import org.bukkit.event.EventHandler;
|
import org.bukkit.event.EventHandler;
|
||||||
import org.bukkit.event.EventPriority;
|
import org.bukkit.event.EventPriority;
|
||||||
import org.bukkit.event.Listener;
|
import org.bukkit.event.Listener;
|
||||||
|
import org.bukkit.event.entity.EntityDamageByEntityEvent;
|
||||||
import org.bukkit.event.inventory.InventoryClickEvent;
|
import org.bukkit.event.inventory.InventoryClickEvent;
|
||||||
|
import org.bukkit.event.player.PlayerArmorStandManipulateEvent;
|
||||||
import org.bukkit.event.player.PlayerInteractAtEntityEvent;
|
import org.bukkit.event.player.PlayerInteractAtEntityEvent;
|
||||||
import org.bukkit.inventory.EquipmentSlot;
|
import org.bukkit.inventory.EquipmentSlot;
|
||||||
import org.bukkit.inventory.ItemStack;
|
import org.bukkit.inventory.ItemStack;
|
||||||
@@ -20,6 +24,41 @@ import java.util.UUID;
|
|||||||
|
|
||||||
public class ASTListener implements Listener {
|
public class ASTListener implements Listener {
|
||||||
|
|
||||||
|
private boolean hasArmorStandBypass(Player player) {
|
||||||
|
return player.isOp() || player.hasPermission("nexuslobby.armorstand.editbypass");
|
||||||
|
}
|
||||||
|
|
||||||
|
@EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true)
|
||||||
|
public void onArmorStandManipulate(PlayerArmorStandManipulateEvent event) {
|
||||||
|
Player player = event.getPlayer();
|
||||||
|
if (hasArmorStandBypass(player)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verhindert, dass Spieler Rüstung/Items aus ArmorStands nehmen oder austauschen.
|
||||||
|
event.setCancelled(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
@EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true)
|
||||||
|
public void onArmorStandDamage(EntityDamageByEntityEvent event) {
|
||||||
|
if (!(event.getEntity() instanceof ArmorStand)) return;
|
||||||
|
|
||||||
|
Player player = null;
|
||||||
|
Entity damager = event.getDamager();
|
||||||
|
if (damager instanceof Player p) {
|
||||||
|
player = p;
|
||||||
|
} else if (damager instanceof Projectile projectile && projectile.getShooter() instanceof Player p) {
|
||||||
|
player = p;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (player != null && hasArmorStandBypass(player)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verhindert Abbauen/Beschädigen ohne Bypass, damit keine Equipment-Drops entnommen werden können.
|
||||||
|
event.setCancelled(true);
|
||||||
|
}
|
||||||
|
|
||||||
@EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true)
|
@EventHandler(priority = EventPriority.HIGHEST, ignoreCancelled = true)
|
||||||
public void onInteract(PlayerInteractAtEntityEvent event) {
|
public void onInteract(PlayerInteractAtEntityEvent event) {
|
||||||
if (event.getHand() != EquipmentSlot.HAND) return;
|
if (event.getHand() != EquipmentSlot.HAND) return;
|
||||||
@@ -119,6 +158,9 @@ public class ASTListener implements Listener {
|
|||||||
if (!title.equals("Nexus ArmorStand Editor") &&
|
if (!title.equals("Nexus ArmorStand Editor") &&
|
||||||
!title.equals("Pose: Körperteil wählen") &&
|
!title.equals("Pose: Körperteil wählen") &&
|
||||||
!title.startsWith("Achsen:")) return;
|
!title.startsWith("Achsen:")) return;
|
||||||
|
|
||||||
|
// Amboss-GUI wird von AnvilRenameGUI selbst verwaltet — hier nicht anfassen
|
||||||
|
if (title.equals("§6Name ändern")) return;
|
||||||
|
|
||||||
event.setCancelled(true);
|
event.setCancelled(true);
|
||||||
if (!(event.getWhoClicked() instanceof Player p)) return;
|
if (!(event.getWhoClicked() instanceof Player p)) return;
|
||||||
@@ -136,8 +178,11 @@ public class ASTListener implements Listener {
|
|||||||
ArmorStandTool tool = ArmorStandTool.get(item);
|
ArmorStandTool tool = ArmorStandTool.get(item);
|
||||||
if (tool != null) {
|
if (tool != null) {
|
||||||
tool.execute(as, p);
|
tool.execute(as, p);
|
||||||
// Menü aktualisieren, falls wir noch im selben Editor sind
|
// Editor nur neu öffnen wenn kein Chat-Input-Modus aktiv ist
|
||||||
if (p.getOpenInventory().getTitle().equals(title)) {
|
// und wir wirklich noch im Editor-Inventar sind
|
||||||
|
String newTitle = p.getOpenInventory().getTitle();
|
||||||
|
if (newTitle.equals("Nexus ArmorStand Editor")
|
||||||
|
&& !AST.chatInputMode.containsKey(p.getUniqueId())) {
|
||||||
new ArmorStandGUI(as, p);
|
new ArmorStandGUI(as, p);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,95 @@
|
|||||||
|
package de.nexuslobby.modules.armorstandtools;
|
||||||
|
|
||||||
|
import de.nexuslobby.NexusLobby;
|
||||||
|
import io.papermc.paper.event.player.AsyncChatEvent;
|
||||||
|
import net.kyori.adventure.text.serializer.plain.PlainTextComponentSerializer;
|
||||||
|
import org.bukkit.Bukkit;
|
||||||
|
import org.bukkit.ChatColor;
|
||||||
|
import org.bukkit.entity.ArmorStand;
|
||||||
|
import org.bukkit.entity.Player;
|
||||||
|
import org.bukkit.event.EventHandler;
|
||||||
|
import org.bukkit.event.EventPriority;
|
||||||
|
import org.bukkit.event.HandlerList;
|
||||||
|
import org.bukkit.event.Listener;
|
||||||
|
import org.bukkit.event.player.PlayerQuitEvent;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Chat-basierte Namenseingabe für ArmorStands.
|
||||||
|
* Der Spieler tippt den neuen Namen in den Chat.
|
||||||
|
* Unterstützt & Farb-Codes.
|
||||||
|
*/
|
||||||
|
public class AnvilRenameGUI implements Listener {
|
||||||
|
|
||||||
|
private final Player player;
|
||||||
|
private final ArmorStand armorStand;
|
||||||
|
|
||||||
|
public AnvilRenameGUI(Player player, ArmorStand armorStand) {
|
||||||
|
this.player = player;
|
||||||
|
this.armorStand = armorStand;
|
||||||
|
|
||||||
|
// Listener registrieren
|
||||||
|
Bukkit.getPluginManager().registerEvents(this, NexusLobby.getInstance());
|
||||||
|
|
||||||
|
// Inventar schließen + Anweisung anzeigen
|
||||||
|
player.closeInventory();
|
||||||
|
|
||||||
|
// Title als UI-Feedback (wie AFK-Anzeige)
|
||||||
|
player.sendTitle(
|
||||||
|
"§6§lName eingeben",
|
||||||
|
"§7Schreibe den neuen Namen in den §eChat§7. §8(&-Codes erlaubt)",
|
||||||
|
10, 80, 20
|
||||||
|
);
|
||||||
|
player.sendMessage("§8[§6Nexus§8] §7Gib den neuen §eNamen §7in den Chat ein. §8(&-Farb-Codes erlaubt)");
|
||||||
|
player.sendMessage("§8[§6Nexus§8] §c'abbrechen' §7eingeben zum Abbrechen.");
|
||||||
|
}
|
||||||
|
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
// Chat-Event — Paper 1.21 AsyncChatEvent
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
@EventHandler(priority = EventPriority.LOWEST)
|
||||||
|
public void onChat(AsyncChatEvent event) {
|
||||||
|
if (!event.getPlayer().equals(player)) return;
|
||||||
|
|
||||||
|
// Chat-Nachricht abfangen
|
||||||
|
event.setCancelled(true);
|
||||||
|
|
||||||
|
// Plain-Text aus Adventure Component extrahieren
|
||||||
|
String input = PlainTextComponentSerializer.plainText().serialize(event.message()).trim();
|
||||||
|
|
||||||
|
// Zurück auf Haupt-Thread wechseln für Bukkit-API Aufrufe
|
||||||
|
Bukkit.getScheduler().runTask(NexusLobby.getInstance(), () -> {
|
||||||
|
HandlerList.unregisterAll(this);
|
||||||
|
AST.chatInputMode.remove(player.getUniqueId());
|
||||||
|
|
||||||
|
if (input.equalsIgnoreCase("abbrechen")) {
|
||||||
|
player.sendTitle("§c§lAbgebrochen", "", 5, 30, 10);
|
||||||
|
player.sendMessage("§8[§6Nexus§8] §cNamensänderung abgebrochen.");
|
||||||
|
} else {
|
||||||
|
// & → § Farb-Codes
|
||||||
|
String newName = ChatColor.translateAlternateColorCodes('&', input);
|
||||||
|
armorStand.setCustomName(newName);
|
||||||
|
armorStand.setCustomNameVisible(!ChatColor.stripColor(newName).isEmpty());
|
||||||
|
|
||||||
|
player.sendTitle("§a§lName gesetzt!", "§r" + newName, 5, 40, 10);
|
||||||
|
player.sendMessage("§8[§6Nexus§8] §aName gesetzt: " + newName);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Zurück zum ArmorStand-Editor
|
||||||
|
if (armorStand.isValid()) {
|
||||||
|
new ArmorStandGUI(armorStand, player);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
// Logout — Listener sauber abmelden
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
@EventHandler
|
||||||
|
public void onQuit(PlayerQuitEvent event) {
|
||||||
|
if (!event.getPlayer().equals(player)) return;
|
||||||
|
HandlerList.unregisterAll(this);
|
||||||
|
AST.chatInputMode.remove(player.getUniqueId());
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -238,6 +238,18 @@ public class ArmorStandCmdExecutor implements CommandExecutor {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (args[0].equalsIgnoreCase("clearbubbles")) {
|
||||||
|
ConversationManager cm = NexusLobby.getInstance().getConversationManager();
|
||||||
|
if (cm == null) {
|
||||||
|
p.sendMessage(prefix + "§cConversationManager ist nicht aktiv.");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
cm.clearHangingBubbles();
|
||||||
|
p.sendMessage(prefix + "§aAlle hängenden Sprechblasen wurden entfernt.");
|
||||||
|
p.playSound(p.getLocation(), org.bukkit.Sound.BLOCK_NOTE_BLOCK_PLING, 1f, 2f);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
return sendHelp(p);
|
return sendHelp(p);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -284,6 +296,7 @@ public class ArmorStandCmdExecutor implements CommandExecutor {
|
|||||||
p.sendMessage("§e/nexuscmd conv §7- Gesprächs-Menü");
|
p.sendMessage("§e/nexuscmd conv §7- Gesprächs-Menü");
|
||||||
p.sendMessage("§e/nexuscmd list §7- Zeigt alle Tags");
|
p.sendMessage("§e/nexuscmd list §7- Zeigt alle Tags");
|
||||||
p.sendMessage("§e/nexuscmd remove §7- Löscht Befehle");
|
p.sendMessage("§e/nexuscmd remove §7- Löscht Befehle");
|
||||||
|
p.sendMessage("§e/nexuscmd clearbubbles §7- Entfernt hängende Sprechblasen");
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -51,9 +51,9 @@ public enum ArmorStandTool {
|
|||||||
case GRAV -> as.setGravity(!as.hasGravity());
|
case GRAV -> as.setGravity(!as.hasGravity());
|
||||||
case INVUL -> as.setInvulnerable(!as.isInvulnerable());
|
case INVUL -> as.setInvulnerable(!as.isInvulnerable());
|
||||||
case SET_NAME -> {
|
case SET_NAME -> {
|
||||||
p.closeInventory();
|
|
||||||
p.sendMessage("§8[§6Nexus§8] §7Nutze §e/nexuscmd name <Text>");
|
|
||||||
AST.selectedArmorStand.put(p.getUniqueId(), as);
|
AST.selectedArmorStand.put(p.getUniqueId(), as);
|
||||||
|
AST.chatInputMode.put(p.getUniqueId(), as);
|
||||||
|
new AnvilRenameGUI(p, as); // schließt Inventar intern
|
||||||
}
|
}
|
||||||
case CONV_SETUP -> {
|
case CONV_SETUP -> {
|
||||||
// Automatisches Markieren via GUI
|
// Automatisches Markieren via GUI
|
||||||
|
|||||||
@@ -23,6 +23,8 @@ public class ConversationManager {
|
|||||||
|
|
||||||
// Verhindert Mehrfach-Gespräche und regelt die 60s Pause
|
// Verhindert Mehrfach-Gespräche und regelt die 60s Pause
|
||||||
private final Set<UUID> activeSpeakers = new HashSet<>();
|
private final Set<UUID> activeSpeakers = new HashSet<>();
|
||||||
|
// Alle aktiven Bubble-Entities tracken – für zuverlässiges Cleanup bei Reload/Crash
|
||||||
|
private final Set<UUID> activeBubbles = new HashSet<>();
|
||||||
|
|
||||||
public ConversationManager(NexusLobby plugin) {
|
public ConversationManager(NexusLobby plugin) {
|
||||||
this.plugin = plugin;
|
this.plugin = plugin;
|
||||||
@@ -50,6 +52,14 @@ public class ConversationManager {
|
|||||||
* zu breit und hätte auch legitime Marker-ArmorStands anderer Systeme gelöscht.
|
* zu breit und hätte auch legitime Marker-ArmorStands anderer Systeme gelöscht.
|
||||||
*/
|
*/
|
||||||
public void clearHangingBubbles() {
|
public void clearHangingBubbles() {
|
||||||
|
// Zuerst: Alle getrackte Bubble-Entities direkt entfernen (schnell, kein World-Scan nötig)
|
||||||
|
for (UUID uuid : new HashSet<>(activeBubbles)) {
|
||||||
|
Entity e = Bukkit.getEntity(uuid);
|
||||||
|
if (e != null) e.remove();
|
||||||
|
}
|
||||||
|
activeBubbles.clear();
|
||||||
|
|
||||||
|
// Danach: Vollständiger World-Scan als Fallback (fängt Bubbles nach Crash/Reload ab)
|
||||||
int count = 0;
|
int count = 0;
|
||||||
for (World world : Bukkit.getWorlds()) {
|
for (World world : Bukkit.getWorlds()) {
|
||||||
for (ArmorStand as : world.getEntitiesByClass(ArmorStand.class)) {
|
for (ArmorStand as : world.getEntitiesByClass(ArmorStand.class)) {
|
||||||
@@ -228,10 +238,15 @@ public class ConversationManager {
|
|||||||
s.setCustomName(ChatColorTranslate(text));
|
s.setCustomName(ChatColorTranslate(text));
|
||||||
s.setCustomNameVisible(true);
|
s.setCustomNameVisible(true);
|
||||||
s.setInvulnerable(true);
|
s.setInvulnerable(true);
|
||||||
|
s.setPersistent(false); // Überlebt keinen Server-Neustart/-Absturz
|
||||||
s.addScoreboardTag("nexus_bubble");
|
s.addScoreboardTag("nexus_bubble");
|
||||||
});
|
});
|
||||||
|
|
||||||
Bukkit.getScheduler().runTaskLater(plugin, bubble::remove, 80L); // 4s Sichtbarkeit
|
activeBubbles.add(bubble.getUniqueId());
|
||||||
|
Bukkit.getScheduler().runTaskLater(plugin, () -> {
|
||||||
|
bubble.remove();
|
||||||
|
activeBubbles.remove(bubble.getUniqueId());
|
||||||
|
}, 80L); // 4s Sichtbarkeit
|
||||||
}
|
}
|
||||||
|
|
||||||
// --- Legacy Support & Config Methoden ---
|
// --- Legacy Support & Config Methoden ---
|
||||||
@@ -275,8 +290,8 @@ public class ConversationManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public void reload() {
|
public void reload() {
|
||||||
setupFile();
|
clearHangingBubbles(); // Erst aufräumen, dann neu laden
|
||||||
activeSpeakers.clear();
|
activeSpeakers.clear();
|
||||||
clearHangingBubbles();
|
setupFile();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -17,6 +17,7 @@ import org.bukkit.event.EventHandler;
|
|||||||
import org.bukkit.event.Listener;
|
import org.bukkit.event.Listener;
|
||||||
import org.bukkit.event.entity.EntityDamageByEntityEvent;
|
import org.bukkit.event.entity.EntityDamageByEntityEvent;
|
||||||
import org.bukkit.event.player.PlayerInteractAtEntityEvent;
|
import org.bukkit.event.player.PlayerInteractAtEntityEvent;
|
||||||
|
import org.bukkit.event.player.PlayerJoinEvent;
|
||||||
import org.bukkit.inventory.ItemStack;
|
import org.bukkit.inventory.ItemStack;
|
||||||
import org.bukkit.inventory.meta.SkullMeta;
|
import org.bukkit.inventory.meta.SkullMeta;
|
||||||
import org.bukkit.profile.PlayerProfile;
|
import org.bukkit.profile.PlayerProfile;
|
||||||
@@ -47,6 +48,9 @@ public class SoccerModule implements Module, Listener, CommandExecutor {
|
|||||||
private static final double VOID_THRESHOLD = -5.0;
|
private static final double VOID_THRESHOLD = -5.0;
|
||||||
private static final double CLEANUP_RADIUS = 5.0;
|
private static final double CLEANUP_RADIUS = 5.0;
|
||||||
|
|
||||||
|
// Verhindert dass Dribbling einen Kick sofort überschreibt.
|
||||||
|
private static final long KICK_DRIBBLE_LOCK_MS = 300L;
|
||||||
|
|
||||||
private static final double PARTICLE_SPEED_HIGH = 0.85;
|
private static final double PARTICLE_SPEED_HIGH = 0.85;
|
||||||
private static final double PARTICLE_SPEED_MEDIUM = 0.45;
|
private static final double PARTICLE_SPEED_MEDIUM = 0.45;
|
||||||
private static final double PARTICLE_SPEED_MIN = 0.05;
|
private static final double PARTICLE_SPEED_MIN = 0.05;
|
||||||
@@ -70,6 +74,7 @@ public class SoccerModule implements Module, Listener, CommandExecutor {
|
|||||||
private Location spawnLocation;
|
private Location spawnLocation;
|
||||||
private long lastMoveTime;
|
private long lastMoveTime;
|
||||||
private long lastGoalTime = 0L;
|
private long lastGoalTime = 0L;
|
||||||
|
private long lastKickTime = 0L;
|
||||||
|
|
||||||
// ── Tor-Erkennung: Position aus dem VORHERIGEN Tick ───────────────────────
|
// ── Tor-Erkennung: Position aus dem VORHERIGEN Tick ───────────────────────
|
||||||
//
|
//
|
||||||
@@ -134,12 +139,11 @@ public class SoccerModule implements Module, Listener, CommandExecutor {
|
|||||||
handleDribbling();
|
handleDribbling();
|
||||||
|
|
||||||
// ── SCHRITT 3: Nach Physik die neue Position lesen ─────────────────
|
// ── SCHRITT 3: Nach Physik die neue Position lesen ─────────────────
|
||||||
// Minecraft hat jetzt die Position um vel verschoben.
|
// Minecraft wendet nach setVelocity noch Drag (0.98 horizontal, Gravity ~0.08
|
||||||
// Da getLocation() einen Tick verzögert ist, lesen wir hier noch
|
// + 0.98 vertikal) an. Wir approximieren: newPos = currentPos + vel * 0.98
|
||||||
// currentPos – aber über den Velocity-Vektor wissen wir die neue Pos.
|
// Das ist deutlich genauer als currentPos + vel und vermeidet 1-Tick-Versatz
|
||||||
// Wir nutzen deshalb: newPos = currentPos + velocity (vor Drag)
|
// bei der Tor-Erkennung nach schnellen Kicks.
|
||||||
// Das ist die tatsächliche neue Ballposition nach diesem Tick.
|
Location newPos = currentPos.clone().add(vel.clone().multiply(0.98));
|
||||||
Location newPos = currentPos.clone().add(vel);
|
|
||||||
|
|
||||||
// ── SCHRITT 4: Segment prüfen ─────────────────────────────────────
|
// ── SCHRITT 4: Segment prüfen ─────────────────────────────────────
|
||||||
// Das Segment prevPos → newPos deckt den vollständigen Weg ab den
|
// Das Segment prevPos → newPos deckt den vollständigen Weg ab den
|
||||||
@@ -277,6 +281,7 @@ public class SoccerModule implements Module, Listener, CommandExecutor {
|
|||||||
|
|
||||||
private void handleDribbling() {
|
private void handleDribbling() {
|
||||||
if (ball == null || !ball.isValid()) return;
|
if (ball == null || !ball.isValid()) return;
|
||||||
|
if (System.currentTimeMillis() - lastKickTime < KICK_DRIBBLE_LOCK_MS) return;
|
||||||
for (Entity nearby : ball.getNearbyEntities(DRIBBLE_DETECTION_RADIUS, DRIBBLE_HEIGHT, DRIBBLE_DETECTION_RADIUS)) {
|
for (Entity nearby : ball.getNearbyEntities(DRIBBLE_DETECTION_RADIUS, DRIBBLE_HEIGHT, DRIBBLE_DETECTION_RADIUS)) {
|
||||||
if (nearby instanceof Player player) {
|
if (nearby instanceof Player player) {
|
||||||
Vector dir = ball.getLocation().toVector().subtract(player.getLocation().toVector());
|
Vector dir = ball.getLocation().toVector().subtract(player.getLocation().toVector());
|
||||||
@@ -290,8 +295,16 @@ public class SoccerModule implements Module, Listener, CommandExecutor {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void handleWallBounce(Vector vel) {
|
private void handleWallBounce(Vector vel) {
|
||||||
if (vel.lengthSquared() < 0.001) return;
|
// Nur prüfen wenn genug horizontale Bewegung vorhanden ist.
|
||||||
|
// Y-Komponente absichtlich ignoriert – der Bounce gilt nur für Wände (X/Z),
|
||||||
|
// nicht für Boden/Decke. So verhindert man dass liegende Boden-Blöcke
|
||||||
|
// fälschlicherweise als Wand erkannt werden und den Ball dauerhaft blockieren.
|
||||||
|
double hSpeed = vel.getX() * vel.getX() + vel.getZ() * vel.getZ();
|
||||||
|
if (hSpeed < 0.001) return;
|
||||||
|
|
||||||
Location loc = ball.getLocation();
|
Location loc = ball.getLocation();
|
||||||
|
// Y + 0.5 anheben damit wir auf Wand-Höhe prüfen, nicht auf Boden-Ebene.
|
||||||
|
// Ohne den Offset würde loc.getY() (Boden des ArmorStands) den Bodenblock selbst treffen.
|
||||||
Block nx = loc.clone().add(vel.getX() * WALL_CHECK_DISTANCE, 0.5, 0).getBlock();
|
Block nx = loc.clone().add(vel.getX() * WALL_CHECK_DISTANCE, 0.5, 0).getBlock();
|
||||||
Block nz = loc.clone().add(0, 0.5, vel.getZ() * WALL_CHECK_DISTANCE).getBlock();
|
Block nz = loc.clone().add(0, 0.5, vel.getZ() * WALL_CHECK_DISTANCE).getBlock();
|
||||||
boolean bounced = false;
|
boolean bounced = false;
|
||||||
@@ -437,7 +450,7 @@ public class SoccerModule implements Module, Listener, CommandExecutor {
|
|||||||
ball.setInvisible(true); ball.setGravity(true); ball.setBasePlate(false);
|
ball.setInvisible(true); ball.setGravity(true); ball.setBasePlate(false);
|
||||||
ball.setSmall(true); ball.setInvulnerable(false); ball.setArms(false);
|
ball.setSmall(true); ball.setInvulnerable(false); ball.setArms(false);
|
||||||
ball.setCustomNameVisible(false); ball.setCustomName(BALL_NAME);
|
ball.setCustomNameVisible(false); ball.setCustomName(BALL_NAME);
|
||||||
ball.setPersistent(false); ball.addScoreboardTag(BALL_TAG);
|
ball.setPersistent(true); ball.addScoreboardTag(BALL_TAG);
|
||||||
if (ball.getEquipment() != null) ball.getEquipment().setHelmet(getSoccerHead());
|
if (ball.getEquipment() != null) ball.getEquipment().setHelmet(getSoccerHead());
|
||||||
prevPos = spawnLocation.clone();
|
prevPos = spawnLocation.clone();
|
||||||
lastMoveTime = System.currentTimeMillis();
|
lastMoveTime = System.currentTimeMillis();
|
||||||
@@ -531,6 +544,7 @@ public class SoccerModule implements Module, Listener, CommandExecutor {
|
|||||||
// prevPos auf aktuelle Position setzen bevor der Ball sich bewegt –
|
// prevPos auf aktuelle Position setzen bevor der Ball sich bewegt –
|
||||||
// so startet das nächste Segment am richtigen Punkt.
|
// so startet das nächste Segment am richtigen Punkt.
|
||||||
prevPos = ball.getLocation().clone();
|
prevPos = ball.getLocation().clone();
|
||||||
|
lastKickTime = System.currentTimeMillis();
|
||||||
ball.setVelocity(dir);
|
ball.setVelocity(dir);
|
||||||
|
|
||||||
ball.getWorld().playSound(ball.getLocation(), Sound.ENTITY_ZOMBIE_ATTACK_IRON_DOOR, 0.6f, 1.5f);
|
ball.getWorld().playSound(ball.getLocation(), Sound.ENTITY_ZOMBIE_ATTACK_IRON_DOOR, 0.6f, 1.5f);
|
||||||
@@ -542,6 +556,45 @@ public class SoccerModule implements Module, Listener, CommandExecutor {
|
|||||||
if (ball != null && event.getRightClicked().equals(ball)) event.setCancelled(true);
|
if (ball != null && event.getRightClicked().equals(ball)) event.setCancelled(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@EventHandler
|
||||||
|
public void onPlayerJoin(PlayerJoinEvent event) {
|
||||||
|
// Wenn ein Spieler joinт: Ball-State prüfen und ggf. resynchronisieren.
|
||||||
|
// Passiert z.B. nach Server-Wechsel via BungeeCord: der Chunk wird neu geladen,
|
||||||
|
// alte Ball-Entities tauchen wieder auf, die interne Referenz zeigt evtl. auf
|
||||||
|
// die falsche oder eine ungültige Instanz.
|
||||||
|
Bukkit.getScheduler().runTaskLater(NexusLobby.getInstance(), () -> {
|
||||||
|
if (spawnLocation == null || spawnLocation.getWorld() == null) return;
|
||||||
|
|
||||||
|
// Alle alten nexusball-Entities in der Welt suchen
|
||||||
|
boolean foundValid = false;
|
||||||
|
for (Entity e : spawnLocation.getWorld().getEntities()) {
|
||||||
|
if (!(e instanceof ArmorStand as)) continue;
|
||||||
|
if (!isBallEntity(as)) continue;
|
||||||
|
|
||||||
|
if (ball != null && ball.isValid() && as.getUniqueId().equals(ball.getUniqueId())) {
|
||||||
|
// Referenz ist noch korrekt und gültig – alles gut
|
||||||
|
foundValid = true;
|
||||||
|
} else if (ball == null || !ball.isValid()) {
|
||||||
|
// Unsere Referenz ist verloren – die gefundene Entity übernehmen
|
||||||
|
ball = as;
|
||||||
|
prevPos = as.getLocation().clone();
|
||||||
|
lastMoveTime = System.currentTimeMillis();
|
||||||
|
foundValid = true;
|
||||||
|
Bukkit.getLogger().info("[Soccer] Ball-Referenz nach Join wiederhergestellt.");
|
||||||
|
} else {
|
||||||
|
// Duplikat – entfernen
|
||||||
|
as.remove();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!foundValid) {
|
||||||
|
// Kein Ball in der Welt – neu spawnen
|
||||||
|
respawnBall();
|
||||||
|
Bukkit.getLogger().info("[Soccer] Ball nach Join neu gespawnt.");
|
||||||
|
}
|
||||||
|
}, 20L); // 1s warten damit der Chunk sicher geladen ist
|
||||||
|
}
|
||||||
|
|
||||||
// =========================================================================
|
// =========================================================================
|
||||||
// COMMANDS
|
// COMMANDS
|
||||||
// =========================================================================
|
// =========================================================================
|
||||||
|
|||||||
@@ -1,77 +1,70 @@
|
|||||||
package de.nexuslobby.modules.gadgets;
|
package de.nexuslobby.modules.gadgets;
|
||||||
|
|
||||||
|
import java.util.UUID;
|
||||||
|
import org.bukkit.Bukkit;
|
||||||
import org.bukkit.Location;
|
import org.bukkit.Location;
|
||||||
import org.bukkit.Material;
|
import org.bukkit.Material;
|
||||||
import org.bukkit.entity.ArmorStand;
|
import org.bukkit.entity.ArmorStand;
|
||||||
|
import org.bukkit.entity.Entity;
|
||||||
import org.bukkit.entity.EntityType;
|
import org.bukkit.entity.EntityType;
|
||||||
import org.bukkit.entity.LivingEntity;
|
import org.bukkit.entity.LivingEntity;
|
||||||
import org.bukkit.entity.Player;
|
import org.bukkit.entity.Player;
|
||||||
import org.bukkit.inventory.ItemStack;
|
import org.bukkit.inventory.ItemStack;
|
||||||
import org.bukkit.util.Vector;
|
import org.bukkit.util.Vector;
|
||||||
|
|
||||||
import java.util.UUID;
|
|
||||||
|
|
||||||
public class Balloon {
|
public class Balloon {
|
||||||
|
private final UUID playerUUID;
|
||||||
|
private final LivingEntity balloonEntity;
|
||||||
|
private final ArmorStand headStand;
|
||||||
|
|
||||||
private final UUID playerUUID;
|
public Balloon(Player player, Material balloonMaterial) {
|
||||||
private final LivingEntity balloonEntity;
|
this.playerUUID = player.getUniqueId();
|
||||||
private final ArmorStand headStand;
|
Location loc = player.getLocation().add((double)0.0F, (double)2.0F, (double)0.0F);
|
||||||
|
this.balloonEntity = (LivingEntity)loc.getWorld().spawnEntity(loc, EntityType.PIG);
|
||||||
|
this.balloonEntity.setInvisible(true);
|
||||||
|
this.balloonEntity.setSilent(true);
|
||||||
|
this.balloonEntity.setInvulnerable(true);
|
||||||
|
this.balloonEntity.setGravity(false);
|
||||||
|
this.balloonEntity.setLeashHolder(player);
|
||||||
|
this.balloonEntity.addScoreboardTag("nexus_balloon");
|
||||||
|
this.headStand = (ArmorStand)loc.getWorld().spawnEntity(loc, EntityType.ARMOR_STAND);
|
||||||
|
this.headStand.setVisible(false);
|
||||||
|
this.headStand.setGravity(false);
|
||||||
|
this.headStand.setMarker(true);
|
||||||
|
this.headStand.setHelmet(new ItemStack(balloonMaterial));
|
||||||
|
}
|
||||||
|
|
||||||
public Balloon(Player player, Material balloonMaterial) {
|
public void update() {
|
||||||
this.playerUUID = player.getUniqueId();
|
Player player = Bukkit.getPlayer(this.playerUUID);
|
||||||
Location loc = player.getLocation().add(0, 2, 0);
|
if (player != null && player.isOnline() && this.balloonEntity != null && this.balloonEntity.isValid()) {
|
||||||
|
if (!this.balloonEntity.isLeashed()) {
|
||||||
|
this.remove();
|
||||||
|
} else {
|
||||||
|
Location targetLoc = player.getLocation().clone().add((double)0.0F, (double)2.5F, (double)0.0F);
|
||||||
|
Vector direction = targetLoc.toVector().subtract(this.balloonEntity.getLocation().toVector());
|
||||||
|
double distance = direction.length();
|
||||||
|
if (distance > (double)5.0F) {
|
||||||
|
this.balloonEntity.teleport(targetLoc);
|
||||||
|
} else if (distance > 0.1) {
|
||||||
|
this.balloonEntity.setVelocity(direction.multiply(0.4));
|
||||||
|
}
|
||||||
|
|
||||||
// Das unsichtbare Träger-Entity
|
this.headStand.teleport(this.balloonEntity.getLocation().clone().subtract((double)0.0F, (double)1.5F, (double)0.0F));
|
||||||
this.balloonEntity = (LivingEntity) loc.getWorld().spawnEntity(loc, EntityType.PIG);
|
}
|
||||||
this.balloonEntity.setInvisible(true);
|
} else {
|
||||||
this.balloonEntity.setSilent(true);
|
this.remove();
|
||||||
this.balloonEntity.setInvulnerable(true);
|
}
|
||||||
this.balloonEntity.setGravity(false);
|
}
|
||||||
this.balloonEntity.setLeashHolder(player);
|
|
||||||
|
|
||||||
// Der ArmorStand, der den farbigen Block trägt
|
public void remove() {
|
||||||
this.headStand = (ArmorStand) loc.getWorld().spawnEntity(loc, EntityType.ARMOR_STAND);
|
if (this.balloonEntity != null) {
|
||||||
this.headStand.setVisible(false);
|
this.balloonEntity.setLeashHolder((Entity)null);
|
||||||
this.headStand.setGravity(false);
|
this.balloonEntity.remove();
|
||||||
this.headStand.setMarker(true);
|
}
|
||||||
|
|
||||||
// Hier wird das übergebene Material gesetzt (z.B. RED_WOOL, BLUE_WOOL etc.)
|
|
||||||
this.headStand.setHelmet(new ItemStack(balloonMaterial));
|
|
||||||
}
|
|
||||||
|
|
||||||
public void update() {
|
if (this.headStand != null) {
|
||||||
Player player = org.bukkit.Bukkit.getPlayer(playerUUID);
|
this.headStand.remove();
|
||||||
|
}
|
||||||
|
|
||||||
if (player == null || !player.isOnline() || balloonEntity == null || !balloonEntity.isValid()) {
|
}
|
||||||
remove();
|
}
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!balloonEntity.isLeashed()) {
|
|
||||||
remove();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
Location targetLoc = player.getLocation().clone().add(0, 2.5, 0);
|
|
||||||
Vector direction = targetLoc.toVector().subtract(balloonEntity.getLocation().toVector());
|
|
||||||
double distance = direction.length();
|
|
||||||
|
|
||||||
if (distance > 5) {
|
|
||||||
balloonEntity.teleport(targetLoc);
|
|
||||||
} else if (distance > 0.1) {
|
|
||||||
balloonEntity.setVelocity(direction.multiply(0.4));
|
|
||||||
}
|
|
||||||
|
|
||||||
headStand.teleport(balloonEntity.getLocation().clone().subtract(0, 1.5, 0));
|
|
||||||
}
|
|
||||||
|
|
||||||
public void remove() {
|
|
||||||
if (balloonEntity != null) {
|
|
||||||
balloonEntity.setLeashHolder(null);
|
|
||||||
balloonEntity.remove();
|
|
||||||
}
|
|
||||||
if (headStand != null) {
|
|
||||||
headStand.remove();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,6 +1,11 @@
|
|||||||
package de.nexuslobby.modules.gadgets;
|
package de.nexuslobby.modules.gadgets;
|
||||||
|
|
||||||
import de.nexuslobby.NexusLobby;
|
import de.nexuslobby.NexusLobby;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Random;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.UUID;
|
||||||
import org.bukkit.Bukkit;
|
import org.bukkit.Bukkit;
|
||||||
import org.bukkit.Location;
|
import org.bukkit.Location;
|
||||||
import org.bukkit.Particle;
|
import org.bukkit.Particle;
|
||||||
@@ -10,54 +15,40 @@ import org.bukkit.entity.EntityType;
|
|||||||
import org.bukkit.entity.Player;
|
import org.bukkit.entity.Player;
|
||||||
import org.bukkit.scheduler.BukkitRunnable;
|
import org.bukkit.scheduler.BukkitRunnable;
|
||||||
|
|
||||||
import java.util.Random;
|
|
||||||
|
|
||||||
public class ChickenRain {
|
public class ChickenRain {
|
||||||
|
private static final Set<UUID> activeRains = Collections.synchronizedSet(new HashSet());
|
||||||
|
|
||||||
// FIX: Verhindert, dass ein Spieler den Regen mehrfach gleichzeitig starten kann.
|
public static void start(final Player player) {
|
||||||
// Ohne diese Prüfung konnten beliebig viele parallele Tasks gestartet werden,
|
if (!activeRains.contains(player.getUniqueId())) {
|
||||||
// was zu hunderten gespawnten Entities in Sekunden führte.
|
activeRains.add(player.getUniqueId());
|
||||||
private static final java.util.Set<java.util.UUID> activeRains =
|
(new BukkitRunnable() {
|
||||||
java.util.Collections.synchronizedSet(new java.util.HashSet<>());
|
|
||||||
|
|
||||||
public static void start(Player player) {
|
|
||||||
if (activeRains.contains(player.getUniqueId())) return; // bereits aktiv
|
|
||||||
activeRains.add(player.getUniqueId());
|
|
||||||
new BukkitRunnable() {
|
|
||||||
int ticks = 0;
|
int ticks = 0;
|
||||||
final Random random = new Random();
|
final Random random = new Random();
|
||||||
|
|
||||||
@Override
|
|
||||||
public void run() {
|
public void run() {
|
||||||
if (!player.isOnline() || ticks > 100) { // 100 Ticks = 5 Sekunden
|
if (player.isOnline() && this.ticks <= 100) {
|
||||||
activeRains.remove(player.getUniqueId());
|
if (this.ticks % 2 == 0) {
|
||||||
this.cancel();
|
Location spawnLoc = player.getLocation().add((this.random.nextDouble() - (double)0.5F) * (double)4.0F, (double)4.0F, (this.random.nextDouble() - (double)0.5F) * (double)4.0F);
|
||||||
return;
|
Chicken chicken = (Chicken)spawnLoc.getWorld().spawnEntity(spawnLoc, EntityType.CHICKEN);
|
||||||
}
|
chicken.setBaby();
|
||||||
|
chicken.setInvulnerable(true);
|
||||||
// Alle 2 Ticks ein Huhn spawnen
|
Bukkit.getScheduler().runTaskLater(NexusLobby.getInstance(), () -> {
|
||||||
if (ticks % 2 == 0) {
|
|
||||||
Location spawnLoc = player.getLocation().add(
|
|
||||||
(random.nextDouble() - 0.5) * 4,
|
|
||||||
4.0,
|
|
||||||
(random.nextDouble() - 0.5) * 4
|
|
||||||
);
|
|
||||||
|
|
||||||
Chicken chicken = (Chicken) spawnLoc.getWorld().spawnEntity(spawnLoc, EntityType.CHICKEN);
|
|
||||||
chicken.setBaby();
|
|
||||||
chicken.setInvulnerable(true);
|
|
||||||
|
|
||||||
// Nach 1.5 Sekunden "ploppt" das Huhn
|
|
||||||
Bukkit.getScheduler().runTaskLater(NexusLobby.getInstance(), () -> {
|
|
||||||
if (chicken.isValid()) {
|
if (chicken.isValid()) {
|
||||||
chicken.getWorld().spawnParticle(Particle.CLOUD, chicken.getLocation(), 5, 0.2, 0.2, 0.2, 0.1);
|
chicken.getWorld().spawnParticle(Particle.CLOUD, chicken.getLocation(), 5, 0.2, 0.2, 0.2, 0.1);
|
||||||
chicken.getWorld().playSound(chicken.getLocation(), Sound.ENTITY_CHICKEN_EGG, 1.0f, 1.5f);
|
chicken.getWorld().playSound(chicken.getLocation(), Sound.ENTITY_CHICKEN_EGG, 1.0F, 1.5F);
|
||||||
chicken.remove();
|
chicken.remove();
|
||||||
}
|
}
|
||||||
}, 30L);
|
|
||||||
}
|
}, 30L);
|
||||||
ticks++;
|
}
|
||||||
|
|
||||||
|
++this.ticks;
|
||||||
|
} else {
|
||||||
|
ChickenRain.activeRains.remove(player.getUniqueId());
|
||||||
|
this.cancel();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}.runTaskTimer(NexusLobby.getInstance(), 0L, 1L);
|
}).runTaskTimer(NexusLobby.getInstance(), 0L, 1L);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,103 +1,100 @@
|
|||||||
package de.nexuslobby.modules.gadgets;
|
package de.nexuslobby.modules.gadgets;
|
||||||
|
|
||||||
import de.nexuslobby.NexusLobby;
|
import de.nexuslobby.NexusLobby;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.UUID;
|
||||||
import org.bukkit.Location;
|
import org.bukkit.Location;
|
||||||
import org.bukkit.Particle;
|
import org.bukkit.Particle;
|
||||||
import org.bukkit.Sound;
|
import org.bukkit.Sound;
|
||||||
import org.bukkit.entity.Entity;
|
import org.bukkit.entity.Entity;
|
||||||
import org.bukkit.entity.Player;
|
import org.bukkit.entity.Player;
|
||||||
|
import org.bukkit.scheduler.BukkitRunnable;
|
||||||
import org.bukkit.util.Vector;
|
import org.bukkit.util.Vector;
|
||||||
|
|
||||||
import java.util.HashSet;
|
|
||||||
import java.util.Set;
|
|
||||||
import java.util.UUID;
|
|
||||||
|
|
||||||
public class FreezeRay {
|
public class FreezeRay {
|
||||||
|
private static final Set<UUID> frozenPlayers = new HashSet();
|
||||||
|
|
||||||
// FIX: private statt public – Zugriff nur über isFrozen() und unfreeze()
|
public static boolean isFrozen(UUID uuid) {
|
||||||
private static final Set<UUID> frozenPlayers = new HashSet<>();
|
return frozenPlayers.contains(uuid);
|
||||||
|
}
|
||||||
|
|
||||||
public static boolean isFrozen(UUID uuid) { return frozenPlayers.contains(uuid); }
|
public static void unfreeze(UUID uuid) {
|
||||||
|
frozenPlayers.remove(uuid);
|
||||||
|
}
|
||||||
|
|
||||||
/** Beim Disconnect aufrufen, damit kein Ghost-Eintrag bleibt. */
|
public static void shoot(Player shooter) {
|
||||||
public static void unfreeze(UUID uuid) { frozenPlayers.remove(uuid); }
|
Location start = shooter.getEyeLocation();
|
||||||
|
Vector direction = start.getDirection();
|
||||||
|
shooter.getWorld().playSound(start, Sound.ENTITY_SNOW_GOLEM_SHOOT, 1.0F, 1.5F);
|
||||||
|
|
||||||
public static void shoot(Player shooter) {
|
for(double d = (double)0.0F; d < (double)15.0F; d += 0.3) {
|
||||||
Location start = shooter.getEyeLocation();
|
Location point = start.clone().add(direction.clone().multiply(d));
|
||||||
Vector direction = start.getDirection();
|
point.getWorld().spawnParticle(Particle.SNOWFLAKE, point, 1, (double)0.0F, (double)0.0F, (double)0.0F, (double)0.0F);
|
||||||
|
|
||||||
// Sound beim Schießen
|
for(Entity entity : point.getWorld().getNearbyEntities(point, 0.8, 0.8, 0.8)) {
|
||||||
shooter.getWorld().playSound(start, Sound.ENTITY_SNOW_GOLEM_SHOOT, 1.0f, 1.5f);
|
if (entity instanceof Player target) {
|
||||||
|
if (target != shooter) {
|
||||||
|
if (GadgetShield.isProtected(target)) {
|
||||||
|
String reason = GadgetShield.isAdminShielded(target) ? "Dieser Spieler hat Gadget-Schutz aktiviert." : "Dieser Spieler ist gerade im Parkour.";
|
||||||
|
shooter.sendMessage("§8[§6Nexus§8] §7" + reason);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// Wir prüfen in 0.3er Schritten bis zu 15 Blöcke weit
|
applyFreeze(target);
|
||||||
for (double d = 0; d < 15; d += 0.3) {
|
return;
|
||||||
Location point = start.clone().add(direction.clone().multiply(d));
|
}
|
||||||
|
|
||||||
// Partikel-Strahl sichtbar machen
|
|
||||||
point.getWorld().spawnParticle(Particle.SNOWFLAKE, point, 1, 0, 0, 0, 0);
|
|
||||||
|
|
||||||
// Prüfung auf Spieler im Umkreis von 0.8 Blöcken (etwas großzügiger)
|
|
||||||
for (Entity entity : point.getWorld().getNearbyEntities(point, 0.8, 0.8, 0.8)) {
|
|
||||||
if (entity instanceof Player target && target != shooter) {
|
|
||||||
applyFreeze(target);
|
|
||||||
return; // Stoppt den Strahl beim ersten Treffer
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
// Stoppe den Strahl, falls er eine Wand trifft
|
|
||||||
if (point.getBlock().getType().isSolid()) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void applyFreeze(Player target) {
|
if (point.getBlock().getType().isSolid()) {
|
||||||
if (frozenPlayers.contains(target.getUniqueId())) return;
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
frozenPlayers.add(target.getUniqueId());
|
}
|
||||||
|
|
||||||
// Fixiere die Position für den Stasis-Effekt
|
|
||||||
final Location freezeLocation = target.getLocation();
|
|
||||||
|
|
||||||
// Feedback für getroffenen Spieler
|
|
||||||
target.sendMessage("§8[§6Nexus§8] §bDu wurdest eingefroren!");
|
|
||||||
target.getWorld().playSound(target.getLocation(), Sound.BLOCK_GLASS_BREAK, 1.0f, 0.5f);
|
|
||||||
|
|
||||||
new org.bukkit.scheduler.BukkitRunnable() {
|
private static void applyFreeze(final Player target) {
|
||||||
|
if (!frozenPlayers.contains(target.getUniqueId())) {
|
||||||
|
frozenPlayers.add(target.getUniqueId());
|
||||||
|
final Location freezeLocation = target.getLocation();
|
||||||
|
target.sendMessage("§8[§6Nexus§8] §bDu wurdest eingefroren!");
|
||||||
|
target.getWorld().playSound(target.getLocation(), Sound.BLOCK_GLASS_BREAK, 1.0F, 0.5F);
|
||||||
|
(new BukkitRunnable() {
|
||||||
int ticks = 0;
|
int ticks = 0;
|
||||||
@Override
|
|
||||||
public void run() {
|
public void run() {
|
||||||
// Sicherheitscheck: Ist der Spieler noch online?
|
if (target.isOnline() && this.ticks < 60) {
|
||||||
if (!target.isOnline() || ticks >= 60) {
|
if (GadgetShield.isProtected(target)) {
|
||||||
frozenPlayers.remove(target.getUniqueId());
|
FreezeRay.frozenPlayers.remove(target.getUniqueId());
|
||||||
this.cancel();
|
this.cancel();
|
||||||
return;
|
} else {
|
||||||
}
|
target.setVelocity(new Vector(0, 0, 0));
|
||||||
|
Location current = target.getLocation();
|
||||||
// Stasis-Effekt: Bewegung auf 0 setzen und Position fixieren
|
if (current.getX() != freezeLocation.getX() || current.getZ() != freezeLocation.getZ()) {
|
||||||
target.setVelocity(new Vector(0, 0, 0));
|
freezeLocation.setYaw(current.getYaw());
|
||||||
|
freezeLocation.setPitch(current.getPitch());
|
||||||
// Alle 2 Ticks zurückteleportieren, falls er versucht zu laufen
|
target.teleport(freezeLocation);
|
||||||
// (Behält die Blickrichtung des Spielers bei)
|
}
|
||||||
Location current = target.getLocation();
|
|
||||||
if (current.getX() != freezeLocation.getX() || current.getZ() != freezeLocation.getZ()) {
|
Location loc = target.getLocation();
|
||||||
freezeLocation.setYaw(current.getYaw());
|
|
||||||
freezeLocation.setPitch(current.getPitch());
|
for(int i = 0; i < 8; ++i) {
|
||||||
target.teleport(freezeLocation);
|
double angle = (double)i * Math.PI / (double)4.0F;
|
||||||
}
|
double x = Math.cos(angle) * 0.7;
|
||||||
|
double z = Math.sin(angle) * 0.7;
|
||||||
// Optischer Käfig (Ring-Effekt)
|
loc.getWorld().spawnParticle(Particle.SNOWFLAKE, loc.clone().add(x, (double)1.0F, z), 1, (double)0.0F, (double)0.0F, (double)0.0F, (double)0.0F);
|
||||||
Location loc = target.getLocation();
|
loc.getWorld().spawnParticle(Particle.SNOWFLAKE, loc.clone().add(x, 0.2, z), 1, (double)0.0F, (double)0.0F, (double)0.0F, (double)0.0F);
|
||||||
for (int i = 0; i < 8; i++) {
|
}
|
||||||
double angle = i * Math.PI / 4;
|
|
||||||
double x = Math.cos(angle) * 0.7;
|
this.ticks += 2;
|
||||||
double z = Math.sin(angle) * 0.7;
|
}
|
||||||
loc.getWorld().spawnParticle(Particle.SNOWFLAKE, loc.clone().add(x, 1, z), 1, 0, 0, 0, 0);
|
} else {
|
||||||
loc.getWorld().spawnParticle(Particle.SNOWFLAKE, loc.clone().add(x, 0.2, z), 1, 0, 0, 0, 0);
|
FreezeRay.frozenPlayers.remove(target.getUniqueId());
|
||||||
}
|
this.cancel();
|
||||||
|
}
|
||||||
ticks += 2;
|
|
||||||
}
|
}
|
||||||
}.runTaskTimer(NexusLobby.getInstance(), 0L, 2L);
|
}).runTaskTimer(NexusLobby.getInstance(), 0L, 2L);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -0,0 +1,45 @@
|
|||||||
|
package de.nexuslobby.modules.gadgets;
|
||||||
|
|
||||||
|
import de.nexuslobby.NexusLobby;
|
||||||
|
import de.nexuslobby.modules.parkour.ParkourManager;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.UUID;
|
||||||
|
import org.bukkit.Sound;
|
||||||
|
import org.bukkit.entity.Player;
|
||||||
|
|
||||||
|
public class GadgetShield {
|
||||||
|
private static final Set<UUID> shielded = new HashSet();
|
||||||
|
|
||||||
|
public static boolean toggle(Player player) {
|
||||||
|
UUID uuid = player.getUniqueId();
|
||||||
|
if (shielded.contains(uuid)) {
|
||||||
|
shielded.remove(uuid);
|
||||||
|
player.sendMessage("§8[§6Nexus§8] §cGadget-Schutz §7deaktiviert.");
|
||||||
|
player.playSound(player.getLocation(), Sound.BLOCK_NOTE_BLOCK_BASS, 1.0F, 1.0F);
|
||||||
|
return false;
|
||||||
|
} else {
|
||||||
|
shielded.add(uuid);
|
||||||
|
player.sendMessage("§8[§6Nexus§8] §aGadget-Schutz §7aktiviert. §8Du bist nun immun gegen Gadgets.");
|
||||||
|
player.playSound(player.getLocation(), Sound.BLOCK_AMETHYST_BLOCK_CHIME, 1.0F, 1.5F);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean isAdminShielded(Player player) {
|
||||||
|
return shielded.contains(player.getUniqueId());
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean isProtected(Player player) {
|
||||||
|
if (shielded.contains(player.getUniqueId())) {
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
ParkourManager pm = NexusLobby.getInstance().getParkourManager();
|
||||||
|
return pm != null && pm.isIngame(player);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void clear() {
|
||||||
|
shielded.clear();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -6,32 +6,20 @@ import org.bukkit.entity.Player;
|
|||||||
import org.bukkit.util.Vector;
|
import org.bukkit.util.Vector;
|
||||||
|
|
||||||
public class GrapplingHook {
|
public class GrapplingHook {
|
||||||
|
public static void pullPlayer(Player player, Location target) {
|
||||||
|
Location playerLoc = player.getLocation();
|
||||||
|
double distance = target.distance(playerLoc);
|
||||||
|
if (!(distance < (double)2.0F) && !(distance > (double)50.0F)) {
|
||||||
|
Vector v = target.toVector().subtract(playerLoc.toVector());
|
||||||
|
v.multiply(0.3);
|
||||||
|
v.setY(v.getY() * 0.6 + (double)0.5F);
|
||||||
|
if (v.length() > (double)2.5F) {
|
||||||
|
v.normalize().multiply((double)2.5F);
|
||||||
|
}
|
||||||
|
|
||||||
public static void pullPlayer(Player player, Location target) {
|
player.setVelocity(v);
|
||||||
Location playerLoc = player.getLocation();
|
player.playSound(playerLoc, Sound.ENTITY_WIND_CHARGE_WIND_BURST, 1.0F, 1.2F);
|
||||||
|
player.playSound(playerLoc, Sound.ITEM_TRIDENT_RIPTIDE_1, 0.5F, 1.5F);
|
||||||
// Vektor vom Spieler zum Ziel berechnen
|
}
|
||||||
double distance = target.distance(playerLoc);
|
}
|
||||||
|
}
|
||||||
// Wenn das Ziel zu nah oder zu weit weg ist, nichts tun
|
|
||||||
if (distance < 2 || distance > 50) return;
|
|
||||||
|
|
||||||
// Berechnung des Wurfs (Vektor)
|
|
||||||
Vector v = target.toVector().subtract(playerLoc.toVector());
|
|
||||||
|
|
||||||
// Den Vektor normalisieren und skalieren (Stärke des Zugs)
|
|
||||||
v.multiply(0.3); // Basis-Geschwindigkeit
|
|
||||||
v.setY(v.getY() * 0.6 + 0.5); // Etwas mehr Höhe für den Bogen-Effekt
|
|
||||||
|
|
||||||
// Geschwindigkeit begrenzen, damit man nicht aus der Map schießt
|
|
||||||
if (v.length() > 2.5) {
|
|
||||||
v.normalize().multiply(2.5);
|
|
||||||
}
|
|
||||||
|
|
||||||
player.setVelocity(v);
|
|
||||||
|
|
||||||
// Sound-Effekt für das "Ziehen"
|
|
||||||
player.playSound(playerLoc, Sound.ENTITY_WIND_CHARGE_WIND_BURST, 1.0f, 1.2f);
|
|
||||||
player.playSound(playerLoc, Sound.ITEM_TRIDENT_RIPTIDE_1, 0.5f, 1.5f);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -6,21 +6,19 @@ import org.bukkit.inventory.ItemStack;
|
|||||||
import org.bukkit.inventory.meta.ItemMeta;
|
import org.bukkit.inventory.meta.ItemMeta;
|
||||||
|
|
||||||
public class HatManager {
|
public class HatManager {
|
||||||
|
public static void setHat(Player player, Material material, String name) {
|
||||||
|
ItemStack hat = new ItemStack(material);
|
||||||
|
ItemMeta meta = hat.getItemMeta();
|
||||||
|
if (meta != null) {
|
||||||
|
meta.setDisplayName("§6Hut: " + name);
|
||||||
|
hat.setItemMeta(meta);
|
||||||
|
}
|
||||||
|
|
||||||
public static void setHat(Player player, Material material, String name) {
|
player.getInventory().setHelmet(hat);
|
||||||
ItemStack hat = new ItemStack(material);
|
player.sendMessage("§8[§6Nexus§8] §aDu trägst nun: " + name);
|
||||||
ItemMeta meta = hat.getItemMeta();
|
}
|
||||||
if (meta != null) {
|
|
||||||
meta.setDisplayName("§6Hut: " + name);
|
|
||||||
hat.setItemMeta(meta);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Den Gegenstand auf den Kopf setzen
|
|
||||||
player.getInventory().setHelmet(hat);
|
|
||||||
player.sendMessage("§8[§6Nexus§8] §aDu trägst nun: " + name);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void removeHat(Player player) {
|
public static void removeHat(Player player) {
|
||||||
player.getInventory().setHelmet(null);
|
player.getInventory().setHelmet((ItemStack)null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
package de.nexuslobby.modules.gadgets;
|
package de.nexuslobby.modules.gadgets;
|
||||||
|
|
||||||
import de.nexuslobby.NexusLobby;
|
import de.nexuslobby.NexusLobby;
|
||||||
|
import org.bukkit.Bukkit;
|
||||||
import org.bukkit.Location;
|
import org.bukkit.Location;
|
||||||
import org.bukkit.Particle;
|
import org.bukkit.Particle;
|
||||||
import org.bukkit.Sound;
|
import org.bukkit.Sound;
|
||||||
@@ -9,39 +10,45 @@ import org.bukkit.entity.Player;
|
|||||||
import org.bukkit.util.Vector;
|
import org.bukkit.util.Vector;
|
||||||
|
|
||||||
public class MeteorStrike {
|
public class MeteorStrike {
|
||||||
|
public static void launch(Player shooter) {
|
||||||
|
Location target = null;
|
||||||
|
Location start = shooter.getEyeLocation();
|
||||||
|
Vector direction = start.getDirection();
|
||||||
|
|
||||||
public static void launch(Player shooter) {
|
for(double d = (double)0.0F; d < (double)30.0F; d += (double)0.5F) {
|
||||||
Location target = null;
|
Location point = start.clone().add(direction.clone().multiply(d));
|
||||||
Location start = shooter.getEyeLocation();
|
if (point.getBlock().getType().isSolid()) {
|
||||||
Vector direction = start.getDirection();
|
target = point;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
for (double d = 0; d < 30; d += 0.5) {
|
if (target != null) {
|
||||||
Location point = start.clone().add(direction.clone().multiply(d));
|
Location finalTarget = target.clone().add((double)0.0F, (double)0.5F, (double)0.0F);
|
||||||
if (point.getBlock().getType().isSolid()) {
|
finalTarget.getWorld().spawnParticle(Particle.FLAME, finalTarget, 20, (double)0.5F, 0.1, (double)0.5F, 0.05);
|
||||||
target = point;
|
shooter.sendMessage("§8[§6Nexus§8] §cMeteorit im Anflug...");
|
||||||
break;
|
Bukkit.getScheduler().runTaskLater(NexusLobby.getInstance(), () -> {
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (target == null) return;
|
|
||||||
final Location finalTarget = target.clone().add(0, 0.5, 0);
|
|
||||||
|
|
||||||
finalTarget.getWorld().spawnParticle(Particle.FLAME, finalTarget, 20, 0.5, 0.1, 0.5, 0.05);
|
|
||||||
shooter.sendMessage("§8[§6Nexus§8] §cMeteorit im Anflug...");
|
|
||||||
|
|
||||||
org.bukkit.Bukkit.getScheduler().runTaskLater(NexusLobby.getInstance(), () -> {
|
|
||||||
// EXPLOSION_EMITTER ist der moderne Name für HUGE_EXPLOSION
|
|
||||||
finalTarget.getWorld().spawnParticle(Particle.EXPLOSION_EMITTER, finalTarget, 1);
|
finalTarget.getWorld().spawnParticle(Particle.EXPLOSION_EMITTER, finalTarget, 1);
|
||||||
finalTarget.getWorld().spawnParticle(Particle.LAVA, finalTarget, 30, 0.5, 0.5, 0.5, 0.1);
|
finalTarget.getWorld().spawnParticle(Particle.LAVA, finalTarget, 30, (double)0.5F, (double)0.5F, (double)0.5F, 0.1);
|
||||||
finalTarget.getWorld().playSound(finalTarget, Sound.ENTITY_GENERIC_EXPLODE, 1.0f, 0.8f);
|
finalTarget.getWorld().playSound(finalTarget, Sound.ENTITY_GENERIC_EXPLODE, 1.0F, 0.8F);
|
||||||
|
|
||||||
for (Entity entity : finalTarget.getWorld().getNearbyEntities(finalTarget, 4, 4, 4)) {
|
for(Entity entity : finalTarget.getWorld().getNearbyEntities(finalTarget, (double)4.0F, (double)4.0F, (double)4.0F)) {
|
||||||
if (entity instanceof Player p) {
|
if (entity instanceof Player p) {
|
||||||
Vector v = p.getLocation().toVector().subtract(finalTarget.toVector()).normalize().multiply(1.5).setY(0.5);
|
if (GadgetShield.isProtected(p)) {
|
||||||
p.setVelocity(v);
|
if (GadgetShield.isAdminShielded(p)) {
|
||||||
p.sendMessage("§cBUMM!");
|
p.sendMessage("§8[§6Nexus§8] §7Dein Gadget-Schutzschild hat den Meteoriten abgelenkt!");
|
||||||
}
|
} else {
|
||||||
|
p.sendMessage("§8[§6Parkour§8] §7Dein Parkour-Schutzschild hat den Meteoriten abgelenkt!");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Vector v = p.getLocation().toVector().subtract(finalTarget.toVector()).normalize().multiply((double)1.5F).setY((double)0.5F);
|
||||||
|
p.setVelocity(v);
|
||||||
|
p.sendMessage("§cBUMM!");
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}, 30L);
|
|
||||||
}
|
}, 30L);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
package de.nexuslobby.modules.gadgets;
|
package de.nexuslobby.modules.gadgets;
|
||||||
|
|
||||||
import de.nexuslobby.NexusLobby;
|
import de.nexuslobby.NexusLobby;
|
||||||
|
import java.util.Random;
|
||||||
import org.bukkit.Bukkit;
|
import org.bukkit.Bukkit;
|
||||||
import org.bukkit.Location;
|
import org.bukkit.Location;
|
||||||
import org.bukkit.Material;
|
import org.bukkit.Material;
|
||||||
@@ -10,80 +11,65 @@ import org.bukkit.block.Block;
|
|||||||
import org.bukkit.entity.Player;
|
import org.bukkit.entity.Player;
|
||||||
import org.bukkit.util.Vector;
|
import org.bukkit.util.Vector;
|
||||||
|
|
||||||
import java.util.Random;
|
|
||||||
|
|
||||||
public class PaintballGun {
|
public class PaintballGun {
|
||||||
private static final Random random = new Random();
|
private static final Random random = new Random();
|
||||||
|
private static final Material[] COLORS;
|
||||||
// Wir nutzen jetzt Wolle für kräftigere Farben
|
|
||||||
private static final Material[] COLORS = {
|
|
||||||
Material.RED_WOOL, Material.BLUE_WOOL, Material.LIME_WOOL,
|
|
||||||
Material.ORANGE_WOOL, Material.MAGENTA_WOOL, Material.LIGHT_BLUE_WOOL,
|
|
||||||
Material.YELLOW_WOOL, Material.PURPLE_WOOL, Material.PINK_WOOL
|
|
||||||
};
|
|
||||||
|
|
||||||
public static void shoot(Player shooter) {
|
public static void shoot(Player shooter) {
|
||||||
Location start = shooter.getEyeLocation();
|
Location start = shooter.getEyeLocation();
|
||||||
Vector direction = start.getDirection();
|
Vector direction = start.getDirection();
|
||||||
Material randomColor = COLORS[random.nextInt(COLORS.length)];
|
Material randomColor = COLORS[random.nextInt(COLORS.length)];
|
||||||
|
shooter.getWorld().playSound(start, Sound.ENTITY_CHICKEN_EGG, 1.0F, 2.0F);
|
||||||
|
|
||||||
shooter.getWorld().playSound(start, Sound.ENTITY_CHICKEN_EGG, 1.0f, 2.0f);
|
for(double d = (double)0.0F; d < (double)25.0F; d += (double)0.5F) {
|
||||||
|
Location point = start.clone().add(direction.clone().multiply(d));
|
||||||
|
point.getWorld().spawnParticle(Particle.ITEM_SNOWBALL, point, 1, (double)0.0F, (double)0.0F, (double)0.0F, (double)0.0F);
|
||||||
|
Block block = point.getBlock();
|
||||||
|
if (block.getType().isSolid()) {
|
||||||
|
impact(block, randomColor);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
for (double d = 0; d < 25; d += 0.5) {
|
}
|
||||||
Location point = start.clone().add(direction.clone().multiply(d));
|
|
||||||
|
|
||||||
// Flug-Partikel (kleiner Rauch oder Schneeball)
|
|
||||||
point.getWorld().spawnParticle(Particle.ITEM_SNOWBALL, point, 1, 0, 0, 0, 0);
|
|
||||||
|
|
||||||
Block block = point.getBlock();
|
private static void impact(Block centerBlock, Material color) {
|
||||||
if (block.getType().isSolid()) {
|
Location centerLoc = centerBlock.getLocation();
|
||||||
impact(block, randomColor);
|
centerLoc.getWorld().playSound(centerLoc, Sound.ENTITY_SLIME_SQUISH, 1.0F, 1.2F);
|
||||||
break;
|
int radius = 2;
|
||||||
|
|
||||||
|
for(int x = -radius; x <= radius; ++x) {
|
||||||
|
for(int y = -radius; y <= radius; ++y) {
|
||||||
|
for(int z = -radius; z <= radius; ++z) {
|
||||||
|
if ((double)(x * x + y * y + z * z) <= (double)(radius * radius) + (double)0.5F) {
|
||||||
|
Block target = centerLoc.clone().add((double)x, (double)y, (double)z).getBlock();
|
||||||
|
if (target.getType().isSolid()) {
|
||||||
|
applyColor(target, color);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void impact(Block centerBlock, Material color) {
|
}
|
||||||
Location centerLoc = centerBlock.getLocation();
|
|
||||||
centerLoc.getWorld().playSound(centerLoc, Sound.ENTITY_SLIME_SQUISH, 1.0f, 1.2f);
|
|
||||||
|
|
||||||
int radius = 2; // Radius der Farbkugel
|
|
||||||
|
|
||||||
// Wir gehen alle Blöcke im Würfel um den Einschlag durch
|
private static void applyColor(Block block, Material color) {
|
||||||
for (int x = -radius; x <= radius; x++) {
|
Location loc = block.getLocation();
|
||||||
for (int y = -radius; y <= radius; y++) {
|
loc.getWorld().spawnParticle(Particle.BLOCK, loc.clone().add((double)0.5F, (double)0.5F, (double)0.5F), 3, 0.1, 0.1, 0.1, color.createBlockData());
|
||||||
for (int z = -radius; z <= radius; z++) {
|
|
||||||
|
|
||||||
// Berechne Distanz für eine Kugelform (statt Würfel)
|
|
||||||
if (x * x + y * y + z * z <= radius * radius + 0.5) {
|
|
||||||
Block target = centerLoc.clone().add(x, y, z).getBlock();
|
|
||||||
|
|
||||||
// Nur solide Blöcke färben (keine Luft/Gras)
|
|
||||||
if (target.getType().isSolid()) {
|
|
||||||
applyColor(target, color);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void applyColor(Block block, Material color) {
|
for(Player online : Bukkit.getOnlinePlayers()) {
|
||||||
Location loc = block.getLocation();
|
online.sendBlockChange(loc, color.createBlockData());
|
||||||
|
}
|
||||||
// Effekt-Partikel am Block
|
|
||||||
loc.getWorld().spawnParticle(Particle.BLOCK, loc.clone().add(0.5, 0.5, 0.5), 3, 0.1, 0.1, 0.1, color.createBlockData());
|
|
||||||
|
|
||||||
// Block-Änderung an alle senden
|
Bukkit.getScheduler().runTaskLater(NexusLobby.getInstance(), () -> {
|
||||||
for (Player online : Bukkit.getOnlinePlayers()) {
|
for(Player online : Bukkit.getOnlinePlayers()) {
|
||||||
online.sendBlockChange(loc, color.createBlockData());
|
online.sendBlockChange(loc, block.getBlockData());
|
||||||
}
|
}
|
||||||
|
|
||||||
// Nach 10 Sekunden zurücksetzen
|
}, 400L);
|
||||||
Bukkit.getScheduler().runTaskLater(NexusLobby.getInstance(), () -> {
|
}
|
||||||
for (Player online : Bukkit.getOnlinePlayers()) {
|
|
||||||
online.sendBlockChange(loc, block.getBlockData());
|
static {
|
||||||
}
|
COLORS = new Material[]{Material.RED_WOOL, Material.BLUE_WOOL, Material.LIME_WOOL, Material.ORANGE_WOOL, Material.MAGENTA_WOOL, Material.LIGHT_BLUE_WOOL, Material.YELLOW_WOOL, Material.PURPLE_WOOL, Material.PINK_WOOL};
|
||||||
}, 400L);
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|||||||
@@ -5,27 +5,29 @@ import org.bukkit.Particle;
|
|||||||
import org.bukkit.entity.Player;
|
import org.bukkit.entity.Player;
|
||||||
|
|
||||||
public class ParticleEffect {
|
public class ParticleEffect {
|
||||||
private final String type;
|
private final String type;
|
||||||
private double angle = 0;
|
private double angle = (double)0.0F;
|
||||||
|
|
||||||
public ParticleEffect(String type) { this.type = type; }
|
public ParticleEffect(String type) {
|
||||||
|
this.type = type;
|
||||||
|
}
|
||||||
|
|
||||||
public void update(Player player) {
|
public void update(Player player) {
|
||||||
Location loc = player.getLocation();
|
Location loc = player.getLocation();
|
||||||
switch (type.toLowerCase()) {
|
switch (this.type.toLowerCase()) {
|
||||||
case "hearts":
|
case "hearts":
|
||||||
player.getWorld().spawnParticle(Particle.HEART, loc.clone().add(0, 2.2, 0), 1, 0.3, 0.3, 0.3, 0);
|
player.getWorld().spawnParticle(Particle.HEART, loc.clone().add((double)0.0F, 2.2, (double)0.0F), 1, 0.3, 0.3, 0.3, (double)0.0F);
|
||||||
break;
|
break;
|
||||||
case "flames":
|
case "flames":
|
||||||
double x = 0.6 * Math.cos(angle);
|
double x = 0.6 * Math.cos(this.angle);
|
||||||
double z = 0.6 * Math.sin(angle);
|
double z = 0.6 * Math.sin(this.angle);
|
||||||
player.getWorld().spawnParticle(Particle.FLAME, loc.clone().add(x, 0.1, z), 1, 0, 0, 0, 0);
|
player.getWorld().spawnParticle(Particle.FLAME, loc.clone().add(x, 0.1, z), 1, (double)0.0F, (double)0.0F, (double)0.0F, (double)0.0F);
|
||||||
angle += 0.2;
|
this.angle += 0.2;
|
||||||
break;
|
break;
|
||||||
case "cloud":
|
case "cloud":
|
||||||
player.getWorld().spawnParticle(Particle.CLOUD, loc.clone().add(0, 2.5, 0), 2, 0.2, 0.05, 0.2, 0);
|
player.getWorld().spawnParticle(Particle.CLOUD, loc.clone().add((double)0.0F, (double)2.5F, (double)0.0F), 2, 0.2, 0.05, 0.2, (double)0.0F);
|
||||||
player.getWorld().spawnParticle(Particle.FALLING_WATER, loc.clone().add(0, 2.4, 0), 1, 0.1, 0, 0.1, 0);
|
player.getWorld().spawnParticle(Particle.FALLING_WATER, loc.clone().add((double)0.0F, 2.4, (double)0.0F), 1, 0.1, (double)0.0F, 0.1, (double)0.0F);
|
||||||
break;
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,158 +1,234 @@
|
|||||||
package de.nexuslobby.modules.gadgets;
|
package de.nexuslobby.modules.gadgets;
|
||||||
|
|
||||||
import de.nexuslobby.NexusLobby;
|
import de.nexuslobby.NexusLobby;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.UUID;
|
||||||
import org.bukkit.Bukkit;
|
import org.bukkit.Bukkit;
|
||||||
import org.bukkit.Location;
|
import org.bukkit.Location;
|
||||||
|
import org.bukkit.NamespacedKey;
|
||||||
|
import org.bukkit.Registry;
|
||||||
|
import org.bukkit.entity.Ageable;
|
||||||
import org.bukkit.entity.Entity;
|
import org.bukkit.entity.Entity;
|
||||||
import org.bukkit.entity.EntityType;
|
import org.bukkit.entity.EntityType;
|
||||||
|
import org.bukkit.entity.Fox;
|
||||||
import org.bukkit.entity.LivingEntity;
|
import org.bukkit.entity.LivingEntity;
|
||||||
import org.bukkit.entity.Player;
|
import org.bukkit.entity.Player;
|
||||||
import org.bukkit.entity.Tameable;
|
import org.bukkit.entity.Tameable;
|
||||||
|
import org.bukkit.entity.Wolf;
|
||||||
|
import org.bukkit.entity.Fox.Type;
|
||||||
import org.bukkit.event.EventHandler;
|
import org.bukkit.event.EventHandler;
|
||||||
|
import org.bukkit.event.HandlerList;
|
||||||
import org.bukkit.event.Listener;
|
import org.bukkit.event.Listener;
|
||||||
import org.bukkit.event.entity.EntityDamageEvent;
|
import org.bukkit.event.entity.EntityDamageEvent;
|
||||||
import org.bukkit.event.entity.EntityTargetEvent;
|
import org.bukkit.event.entity.EntityTargetEvent;
|
||||||
import org.bukkit.event.player.PlayerQuitEvent;
|
import org.bukkit.event.player.PlayerQuitEvent;
|
||||||
|
import org.bukkit.scoreboard.Scoreboard;
|
||||||
import java.util.HashMap;
|
import org.bukkit.scoreboard.Team;
|
||||||
import java.util.Map;
|
|
||||||
import java.util.UUID;
|
|
||||||
|
|
||||||
public class PetManager implements Listener {
|
public class PetManager implements Listener {
|
||||||
|
private static final Map<UUID, Entity> activePets = new HashMap();
|
||||||
|
private static PetManager instance;
|
||||||
|
|
||||||
private static final Map<UUID, Entity> activePets = new HashMap<>();
|
public PetManager() {
|
||||||
|
Bukkit.getPluginManager().registerEvents(this, NexusLobby.getInstance());
|
||||||
|
}
|
||||||
|
|
||||||
// Singleton-Instanz, damit registerEvents() nur einmal aufgerufen wird
|
public static void register() {
|
||||||
private static PetManager instance;
|
if (instance == null) {
|
||||||
|
instance = new PetManager();
|
||||||
|
}
|
||||||
|
|
||||||
public PetManager() {
|
}
|
||||||
Bukkit.getPluginManager().registerEvents(this, NexusLobby.getInstance());
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
public static void unregister() {
|
||||||
* Registriert den PetManager als Listener, falls noch nicht geschehen.
|
if (instance != null) {
|
||||||
* Muss einmalig beim Plugin-Start aufgerufen werden (z.B. aus GadgetModule.onEnable).
|
HandlerList.unregisterAll(instance);
|
||||||
*/
|
instance = null;
|
||||||
public static void register() {
|
}
|
||||||
if (instance == null) {
|
|
||||||
instance = new PetManager();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void unregister() {
|
}
|
||||||
if (instance != null) {
|
|
||||||
org.bukkit.event.HandlerList.unregisterAll(instance);
|
|
||||||
instance = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
public static void spawnEntityPet(Player player, String type) {
|
||||||
* Spawnt ein echtes Tier-Entity für den Spieler.
|
spawnEntityPet(player, type, false);
|
||||||
*/
|
}
|
||||||
public static void spawnEntityPet(Player player, String type) {
|
|
||||||
removePet(player);
|
|
||||||
|
|
||||||
EntityType entityType;
|
public static void spawnEntityPet(Player player, String type, boolean baby) {
|
||||||
try {
|
spawnEntityPet(player, type, baby, (String)null);
|
||||||
entityType = EntityType.valueOf(type);
|
}
|
||||||
} catch (IllegalArgumentException e) {
|
|
||||||
player.sendMessage("§cFehler: Tier-Typ nicht gefunden.");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
Location loc = player.getLocation();
|
public static void spawnEntityPet(Player player, String type, boolean baby, String variantName) {
|
||||||
Entity pet = player.getWorld().spawnEntity(loc, entityType);
|
removePet(player);
|
||||||
|
|
||||||
pet.setCustomName("§d" + player.getName() + "'s " + capitalize(type.toLowerCase()));
|
EntityType entityType;
|
||||||
pet.setCustomNameVisible(true);
|
try {
|
||||||
pet.setInvulnerable(true);
|
entityType = EntityType.valueOf(type);
|
||||||
pet.setPersistent(false);
|
} catch (IllegalArgumentException var12) {
|
||||||
|
player.sendMessage("§cFehler: Tier-Typ nicht gefunden.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (pet instanceof LivingEntity) {
|
Location loc = player.getLocation();
|
||||||
LivingEntity le = (LivingEntity) pet;
|
Entity pet = player.getWorld().spawnEntity(loc, entityType);
|
||||||
le.setRemoveWhenFarAway(false);
|
String agePrefix = baby ? "§bBaby " : "";
|
||||||
|
String petName = buildPetDisplayName(type, variantName);
|
||||||
// Verhindert, dass das Pet andere angreift
|
pet.setCustomName(agePrefix + petName);
|
||||||
if (le instanceof Tameable) {
|
pet.setCustomNameVisible(true);
|
||||||
((Tameable) le).setTamed(true);
|
pet.setInvulnerable(true);
|
||||||
((Tameable) le).setOwner(player);
|
pet.setPersistent(false);
|
||||||
|
applyNeutralPetTeam(pet);
|
||||||
|
if (pet instanceof LivingEntity le) {
|
||||||
|
le.setRemoveWhenFarAway(false);
|
||||||
|
if (le instanceof Ageable ageable) {
|
||||||
|
if (baby) {
|
||||||
|
ageable.setBaby();
|
||||||
|
} else {
|
||||||
|
ageable.setAdult();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
activePets.put(player.getUniqueId(), pet);
|
if (le instanceof Wolf && variantName != null && !variantName.isEmpty()) {
|
||||||
}
|
Wolf.Variant variant = resolveWolfVariant(variantName);
|
||||||
|
if (variant != null) {
|
||||||
|
((Wolf)le).setVariant(variant);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
if (le instanceof Fox && variantName != null && !variantName.isEmpty()) {
|
||||||
* Steuert das Folgen der Tiere. Wird vom GadgetModule-Timer aufgerufen.
|
try {
|
||||||
*/
|
((Fox)le).setFoxType(Type.valueOf(variantName));
|
||||||
public static void updatePets() {
|
} catch (IllegalArgumentException var11) {
|
||||||
for (Map.Entry<UUID, Entity> entry : activePets.entrySet()) {
|
}
|
||||||
Player owner = Bukkit.getPlayer(entry.getKey());
|
}
|
||||||
Entity pet = entry.getValue();
|
|
||||||
|
|
||||||
if (owner == null || !owner.isOnline() || pet.isDead()) {
|
if (le instanceof Tameable) {
|
||||||
continue;
|
((Tameable)le).setTamed(true);
|
||||||
|
((Tameable)le).setOwner(player);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
activePets.put(player.getUniqueId(), pet);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void updatePets() {
|
||||||
|
for(Map.Entry<UUID, Entity> entry : activePets.entrySet()) {
|
||||||
|
Player owner = Bukkit.getPlayer((UUID)entry.getKey());
|
||||||
|
Entity pet = (Entity)entry.getValue();
|
||||||
|
if (owner != null && owner.isOnline() && !pet.isDead()) {
|
||||||
|
if (pet.getWorld().equals(owner.getWorld()) && !(pet.getLocation().distance(owner.getLocation()) > (double)10.0F)) {
|
||||||
|
if (pet.getLocation().distance(owner.getLocation()) > (double)3.0F) {
|
||||||
|
Location target = owner.getLocation().clone().add(owner.getLocation().getDirection().multiply((double)-1.5F));
|
||||||
|
target.setY(owner.getLocation().getY());
|
||||||
|
pet.teleport(pet.getLocation().add(target.toVector().subtract(pet.getLocation().toVector()).normalize().multiply(0.2)));
|
||||||
|
Location lookAt = pet.getLocation();
|
||||||
|
lookAt.setDirection(owner.getLocation().toVector().subtract(pet.getLocation().toVector()));
|
||||||
|
pet.teleport(lookAt);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
pet.teleport(owner.getLocation());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void removePet(Player player) {
|
||||||
|
if (activePets.containsKey(player.getUniqueId())) {
|
||||||
|
Entity pet = (Entity)activePets.get(player.getUniqueId());
|
||||||
|
removeFromPetTeam(pet);
|
||||||
|
pet.remove();
|
||||||
|
activePets.remove(player.getUniqueId());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void clearAll() {
|
||||||
|
for(Entity pet : activePets.values()) {
|
||||||
|
removeFromPetTeam(pet);
|
||||||
|
pet.remove();
|
||||||
|
}
|
||||||
|
|
||||||
|
activePets.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static String capitalize(String str) {
|
||||||
|
if (str != null && !str.isEmpty()) {
|
||||||
|
String var10000 = str.substring(0, 1).toUpperCase();
|
||||||
|
return var10000 + str.substring(1);
|
||||||
|
} else {
|
||||||
|
return str;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static String buildPetDisplayName(String type, String variantName) {
|
||||||
|
String baseType = capitalize(type.toLowerCase());
|
||||||
|
if (variantName != null && !variantName.isEmpty()) {
|
||||||
|
String variant = capitalize(variantName.toLowerCase());
|
||||||
|
return variant + " " + baseType;
|
||||||
|
} else {
|
||||||
|
return baseType;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Wolf.Variant resolveWolfVariant(String variantName) {
|
||||||
|
String keyName = variantName.toLowerCase();
|
||||||
|
NamespacedKey key = NamespacedKey.minecraft(keyName);
|
||||||
|
return (Wolf.Variant)Registry.WOLF_VARIANT.get(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Team getOrCreatePetTeam() {
|
||||||
|
Scoreboard scoreboard = Bukkit.getScoreboardManager().getMainScoreboard();
|
||||||
|
Team team = scoreboard.getTeam("nexus_pets");
|
||||||
|
if (team == null) {
|
||||||
|
team = scoreboard.registerNewTeam("nexus_pets");
|
||||||
|
team.setPrefix("");
|
||||||
|
team.setSuffix("");
|
||||||
|
}
|
||||||
|
|
||||||
|
return team;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void applyNeutralPetTeam(Entity pet) {
|
||||||
|
Team team = getOrCreatePetTeam();
|
||||||
|
String entry = pet.getUniqueId().toString();
|
||||||
|
if (!team.hasEntry(entry)) {
|
||||||
|
team.addEntry(entry);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void removeFromPetTeam(Entity pet) {
|
||||||
|
if (pet != null) {
|
||||||
|
Scoreboard scoreboard = Bukkit.getScoreboardManager().getMainScoreboard();
|
||||||
|
Team team = scoreboard.getTeam("nexus_pets");
|
||||||
|
if (team != null) {
|
||||||
|
String entry = pet.getUniqueId().toString();
|
||||||
|
if (team.hasEntry(entry)) {
|
||||||
|
team.removeEntry(entry);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Wenn das Pet in einer anderen Welt ist oder zu weit weg, teleportiere es
|
}
|
||||||
if (!pet.getWorld().equals(owner.getWorld()) || pet.getLocation().distance(owner.getLocation()) > 10) {
|
}
|
||||||
pet.teleport(owner.getLocation());
|
}
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Sanftes Folgen: Wenn das Pet weiter als 3 Blöcke weg ist
|
@EventHandler
|
||||||
if (pet.getLocation().distance(owner.getLocation()) > 3) {
|
public void onPetDamage(EntityDamageEvent event) {
|
||||||
Location target = owner.getLocation().clone().add(owner.getLocation().getDirection().multiply(-1.5));
|
if (activePets.containsValue(event.getEntity())) {
|
||||||
target.setY(owner.getLocation().getY());
|
event.setCancelled(true);
|
||||||
|
}
|
||||||
// Teleportiert das Pet leicht zum Ziel (simuliert Laufen)
|
|
||||||
pet.teleport(pet.getLocation().add(target.toVector().subtract(pet.getLocation().toVector()).normalize().multiply(0.2)));
|
|
||||||
|
|
||||||
// Blickrichtung anpassen
|
|
||||||
Location lookAt = pet.getLocation();
|
|
||||||
lookAt.setDirection(owner.getLocation().toVector().subtract(pet.getLocation().toVector()));
|
|
||||||
pet.teleport(lookAt);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void removePet(Player player) {
|
}
|
||||||
if (activePets.containsKey(player.getUniqueId())) {
|
|
||||||
activePets.get(player.getUniqueId()).remove();
|
|
||||||
activePets.remove(player.getUniqueId());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void clearAll() {
|
@EventHandler
|
||||||
for (Entity pet : activePets.values()) {
|
public void onPetTarget(EntityTargetEvent event) {
|
||||||
pet.remove();
|
if (activePets.containsValue(event.getEntity())) {
|
||||||
}
|
event.setCancelled(true);
|
||||||
activePets.clear();
|
}
|
||||||
}
|
|
||||||
|
|
||||||
private static String capitalize(String str) {
|
}
|
||||||
if (str == null || str.isEmpty()) return str;
|
|
||||||
return str.substring(0, 1).toUpperCase() + str.substring(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
// --- Events um die Pets zu schützen ---
|
@EventHandler
|
||||||
|
public void onQuit(PlayerQuitEvent event) {
|
||||||
@EventHandler
|
removePet(event.getPlayer());
|
||||||
public void onPetDamage(EntityDamageEvent event) {
|
}
|
||||||
if (activePets.containsValue(event.getEntity())) {
|
}
|
||||||
event.setCancelled(true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@EventHandler
|
|
||||||
public void onPetTarget(EntityTargetEvent event) {
|
|
||||||
if (activePets.containsValue(event.getEntity())) {
|
|
||||||
event.setCancelled(true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@EventHandler
|
|
||||||
public void onQuit(PlayerQuitEvent event) {
|
|
||||||
removePet(event.getPlayer());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,5 +1,8 @@
|
|||||||
package de.nexuslobby.modules.gadgets;
|
package de.nexuslobby.modules.gadgets;
|
||||||
|
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.UUID;
|
||||||
import org.bukkit.Particle;
|
import org.bukkit.Particle;
|
||||||
import org.bukkit.Sound;
|
import org.bukkit.Sound;
|
||||||
import org.bukkit.entity.Entity;
|
import org.bukkit.entity.Entity;
|
||||||
@@ -7,26 +10,32 @@ import org.bukkit.entity.Player;
|
|||||||
import org.bukkit.util.Vector;
|
import org.bukkit.util.Vector;
|
||||||
|
|
||||||
public class ShieldTask {
|
public class ShieldTask {
|
||||||
|
private static final Map<UUID, Long> pushCooldowns = new HashMap();
|
||||||
|
private static final long PUSH_COOLDOWN_MS = 3000L;
|
||||||
|
|
||||||
public static void handleShield(Player owner) {
|
public static void handleShield(Player owner) {
|
||||||
// Erzeuge einen Partikel-Ring um den Spieler
|
for(double i = (double)0.0F; i < (Math.PI * 2D); i += (Math.PI / 8D)) {
|
||||||
for (double i = 0; i < Math.PI * 2; i += Math.PI / 8) {
|
double x = Math.cos(i) * 2.2;
|
||||||
double x = Math.cos(i) * 2.2;
|
double z = Math.sin(i) * 2.2;
|
||||||
double z = Math.sin(i) * 2.2;
|
owner.getWorld().spawnParticle(Particle.WITCH, owner.getLocation().add(x, (double)0.5F, z), 1, (double)0.0F, (double)0.0F, (double)0.0F, (double)0.0F);
|
||||||
owner.getWorld().spawnParticle(Particle.WITCH, owner.getLocation().add(x, 0.5, z), 1, 0, 0, 0, 0);
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// Stoße andere Spieler weg
|
long now = System.currentTimeMillis();
|
||||||
for (Entity entity : owner.getNearbyEntities(2.2, 2.0, 2.2)) {
|
|
||||||
if (entity instanceof Player && entity != owner) {
|
for(Entity entity : owner.getNearbyEntities(2.2, (double)2.0F, 2.2)) {
|
||||||
Player target = (Player) entity;
|
if (entity instanceof Player target) {
|
||||||
|
if (entity != owner && !GadgetShield.isProtected(target)) {
|
||||||
Vector direction = target.getLocation().toVector().subtract(owner.getLocation().toVector()).normalize();
|
long lastPush = (Long)pushCooldowns.getOrDefault(target.getUniqueId(), 0L);
|
||||||
direction.multiply(0.4).setY(0.2);
|
if (now - lastPush >= 3000L) {
|
||||||
|
pushCooldowns.put(target.getUniqueId(), now);
|
||||||
target.setVelocity(direction);
|
Vector direction = target.getLocation().toVector().subtract(owner.getLocation().toVector()).normalize();
|
||||||
target.playSound(target.getLocation(), Sound.ENTITY_CHICKEN_EGG, 0.5f, 0.5f);
|
direction.multiply(0.4).setY(0.2);
|
||||||
|
target.setVelocity(direction);
|
||||||
|
target.playSound(target.getLocation(), Sound.ENTITY_CHICKEN_EGG, 0.5F, 0.5F);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -68,6 +68,10 @@ public class HoloCommand implements CommandExecutor {
|
|||||||
}
|
}
|
||||||
module.removeHologram(args[1]);
|
module.removeHologram(args[1]);
|
||||||
player.sendMessage("§8[§6Nexus§8] §cHologramm §e" + args[1] + " §agelöscht.");
|
player.sendMessage("§8[§6Nexus§8] §cHologramm §e" + args[1] + " §agelöscht.");
|
||||||
|
}
|
||||||
|
else if (args[0].equalsIgnoreCase("reload")) {
|
||||||
|
module.reloadHolograms();
|
||||||
|
player.sendMessage("§8[§6Nexus§8] §aHologramme neu geladen.");
|
||||||
} else {
|
} else {
|
||||||
sendHelp(player);
|
sendHelp(player);
|
||||||
}
|
}
|
||||||
@@ -79,6 +83,7 @@ public class HoloCommand implements CommandExecutor {
|
|||||||
player.sendMessage("§8§m-----------§r §6Hologramme §8§m-----------");
|
player.sendMessage("§8§m-----------§r §6Hologramme §8§m-----------");
|
||||||
player.sendMessage("§e/holo create <id> <NONE|FAST|SLOW> <text>");
|
player.sendMessage("§e/holo create <id> <NONE|FAST|SLOW> <text>");
|
||||||
player.sendMessage("§e/holo delete <id>");
|
player.sendMessage("§e/holo delete <id>");
|
||||||
|
player.sendMessage("§e/holo reload");
|
||||||
player.sendMessage("§7Nutze §b; §7für neue Seiten.");
|
player.sendMessage("§7Nutze §b; §7für neue Seiten.");
|
||||||
player.sendMessage("§7Nutze §b\\n §7für Zeilenumbruch.");
|
player.sendMessage("§7Nutze §b\\n §7für Zeilenumbruch.");
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -153,15 +153,57 @@ public class HologramModule implements Module, Listener {
|
|||||||
saveHoloConfig();
|
saveHoloConfig();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Lädt die holograms.yml neu und rendert alle Hologramme für Online-Spieler neu.
|
||||||
|
* Wird von HoloCommand (/holo reload) aufgerufen.
|
||||||
|
*/
|
||||||
|
public void reloadHolograms() {
|
||||||
|
// Alle aktiven Hologramme für alle Spieler entfernen
|
||||||
|
for (Player player : Bukkit.getOnlinePlayers()) {
|
||||||
|
holograms.values().forEach(h -> h.removeForPlayer(player));
|
||||||
|
}
|
||||||
|
// Config neu einlesen und Hologramme neu laden
|
||||||
|
loadConfig();
|
||||||
|
loadHolograms();
|
||||||
|
}
|
||||||
|
|
||||||
private void saveHoloConfig() {
|
private void saveHoloConfig() {
|
||||||
try {
|
try {
|
||||||
config.save(file);
|
// Sicherstellen, dass Placeholder-Strings (z.B. %bungeetotal%) korrekt
|
||||||
} catch (IOException e) {
|
// gequotet gespeichert werden, damit SnakeYAML sie nicht als YAML-Direktive
|
||||||
|
// missinterpretiert. Dazu speichern wir die text-Listen explizit als
|
||||||
|
// String-Werte mit einfachen Anführungszeichen via eigener Serialisierung.
|
||||||
|
StringBuilder yaml = new StringBuilder();
|
||||||
|
for (String id : config.getKeys(false)) {
|
||||||
|
yaml.append(id).append(":\n");
|
||||||
|
yaml.append(" world: ").append(quoteIfNeeded(config.getString(id + ".world", ""))).append("\n");
|
||||||
|
yaml.append(" x: ").append(config.getDouble(id + ".x")).append("\n");
|
||||||
|
yaml.append(" y: ").append(config.getDouble(id + ".y")).append("\n");
|
||||||
|
yaml.append(" z: ").append(config.getDouble(id + ".z")).append("\n");
|
||||||
|
yaml.append(" text:\n");
|
||||||
|
for (String line : config.getStringList(id + ".text")) {
|
||||||
|
yaml.append(" - ").append(quoteIfNeeded(line)).append("\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
java.nio.file.Files.writeString(file.toPath(), yaml.toString());
|
||||||
|
} catch (IOException e) {
|
||||||
NexusLobby.getInstance().getLogger().severe("Konnte holograms.yml nicht speichern!");
|
NexusLobby.getInstance().getLogger().severe("Konnte holograms.yml nicht speichern!");
|
||||||
NexusLobby.getInstance().getLogger().severe("Fehler beim Speichern der Hologramme: " + e.getMessage());
|
NexusLobby.getInstance().getLogger().severe("Fehler beim Speichern der Hologramme: " + e.getMessage());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Umschließt Strings mit einfachen Anführungszeichen, wenn sie YAML-Sonderzeichen
|
||||||
|
* wie '%', ':', '#' enthalten, um Parsing-Fehler zu vermeiden.
|
||||||
|
*/
|
||||||
|
private String quoteIfNeeded(String value) {
|
||||||
|
if (value == null) return "''";
|
||||||
|
if (value.contains("%") || value.contains(":") || value.contains("#") || value.contains("'")) {
|
||||||
|
return "'" + value.replace("'", "''") + "'";
|
||||||
|
}
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* WICHTIG: Diese Methode wird vom LobbyTabCompleter benötigt!
|
* WICHTIG: Diese Methode wird vom LobbyTabCompleter benötigt!
|
||||||
* @return Set aller registrierten Hologramm-IDs
|
* @return Set aller registrierten Hologramm-IDs
|
||||||
|
|||||||
@@ -2,8 +2,6 @@ package de.nexuslobby.modules.intro;
|
|||||||
|
|
||||||
import de.nexuslobby.NexusLobby;
|
import de.nexuslobby.NexusLobby;
|
||||||
import de.nexuslobby.api.Module;
|
import de.nexuslobby.api.Module;
|
||||||
import net.md_5.bungee.api.ChatMessageType;
|
|
||||||
import net.md_5.bungee.api.chat.TextComponent;
|
|
||||||
import org.bukkit.Bukkit;
|
import org.bukkit.Bukkit;
|
||||||
import org.bukkit.GameMode;
|
import org.bukkit.GameMode;
|
||||||
import org.bukkit.Location;
|
import org.bukkit.Location;
|
||||||
@@ -16,7 +14,6 @@ import org.bukkit.entity.Player;
|
|||||||
import org.bukkit.event.EventHandler;
|
import org.bukkit.event.EventHandler;
|
||||||
import org.bukkit.event.Listener;
|
import org.bukkit.event.Listener;
|
||||||
import org.bukkit.event.player.PlayerJoinEvent;
|
import org.bukkit.event.player.PlayerJoinEvent;
|
||||||
import org.bukkit.event.player.PlayerToggleSneakEvent;
|
|
||||||
import org.bukkit.scheduler.BukkitRunnable;
|
import org.bukkit.scheduler.BukkitRunnable;
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
|
||||||
@@ -27,13 +24,24 @@ import java.util.*;
|
|||||||
public class IntroModule implements Module, Listener, CommandExecutor {
|
public class IntroModule implements Module, Listener, CommandExecutor {
|
||||||
|
|
||||||
private final Set<UUID> activeIntro = new HashSet<>();
|
private final Set<UUID> activeIntro = new HashSet<>();
|
||||||
|
|
||||||
private final List<Location> points = new ArrayList<>();
|
private final List<Location> points = new ArrayList<>();
|
||||||
|
private final List<String> labels = new ArrayList<>();
|
||||||
|
|
||||||
|
// Abschluss-Titel (konfigurierbar)
|
||||||
|
private String endTitle = "§a§lWillkommen!";
|
||||||
|
private String endSubtitle = "§7Viel Spaß auf NexusLobby";
|
||||||
|
|
||||||
private File configFile;
|
private File configFile;
|
||||||
private FileConfiguration config;
|
private FileConfiguration config;
|
||||||
|
|
||||||
// --- Einstellungen ---
|
// --- Einstellungen ---
|
||||||
private final int TICKS_FLUG = 70; // Dauer der Fahrt zwischen zwei Punkten (3.5 Sek)
|
private final int TICKS_FLUG = 70;
|
||||||
private final int TICKS_PAUSE = 30; // Standzeit an jedem Punkt (1.5 Sek)
|
private final int TICKS_PAUSE = 30;
|
||||||
|
|
||||||
|
private final int TITLE_FADE_IN = 10;
|
||||||
|
private final int TITLE_STAY = 50;
|
||||||
|
private final int TITLE_FADE_OUT = 10;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String getName() { return "Intro"; }
|
public String getName() { return "Intro"; }
|
||||||
@@ -44,7 +52,7 @@ public class IntroModule implements Module, Listener, CommandExecutor {
|
|||||||
if (NexusLobby.getInstance().getCommand("intro") != null) {
|
if (NexusLobby.getInstance().getCommand("intro") != null) {
|
||||||
NexusLobby.getInstance().getCommand("intro").setExecutor(this);
|
NexusLobby.getInstance().getCommand("intro").setExecutor(this);
|
||||||
}
|
}
|
||||||
loadPoints();
|
loadConfig();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -53,30 +61,79 @@ public class IntroModule implements Module, Listener, CommandExecutor {
|
|||||||
activeIntro.clear();
|
activeIntro.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void loadPoints() {
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
// Persistenz
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
private void loadConfig() {
|
||||||
points.clear();
|
points.clear();
|
||||||
|
labels.clear();
|
||||||
configFile = new File(NexusLobby.getInstance().getDataFolder(), "intro.yml");
|
configFile = new File(NexusLobby.getInstance().getDataFolder(), "intro.yml");
|
||||||
if (!configFile.exists()) {
|
if (!configFile.exists()) {
|
||||||
try { configFile.createNewFile(); } catch (IOException ignored) {}
|
try { configFile.createNewFile(); } catch (IOException ignored) {}
|
||||||
}
|
}
|
||||||
config = YamlConfiguration.loadConfiguration(configFile);
|
config = YamlConfiguration.loadConfiguration(configFile);
|
||||||
List<?> list = config.getList("points");
|
|
||||||
|
// Abschluss-Titel laden
|
||||||
|
endTitle = config.getString("end-title", "§a§lWillkommen!");
|
||||||
|
endSubtitle = config.getString("end-subtitle", "§7Viel Spaß auf NexusLobby");
|
||||||
|
|
||||||
|
// Punkte laden
|
||||||
|
List<?> list = config.getList("intro-points");
|
||||||
if (list != null) {
|
if (list != null) {
|
||||||
for (Object obj : list) {
|
for (Object obj : list) {
|
||||||
if (obj instanceof Location loc) points.add(loc);
|
if (obj instanceof Map<?, ?> map) {
|
||||||
|
Object locObj = map.get("location");
|
||||||
|
Object labelObj = map.get("label");
|
||||||
|
if (locObj instanceof Location loc) {
|
||||||
|
points.add(loc);
|
||||||
|
labels.add(labelObj != null ? labelObj.toString() : "");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Rückwärts-Kompatibilität
|
||||||
|
if (points.isEmpty()) {
|
||||||
|
List<?> oldList = config.getList("points");
|
||||||
|
if (oldList != null) {
|
||||||
|
for (Object obj : oldList) {
|
||||||
|
if (obj instanceof Location loc) {
|
||||||
|
points.add(loc);
|
||||||
|
labels.add("");
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void savePoints() {
|
private void saveConfig() {
|
||||||
config.set("points", points);
|
// Abschluss-Titel speichern
|
||||||
try {
|
config.set("end-title", endTitle);
|
||||||
config.save(configFile);
|
config.set("end-subtitle", endSubtitle);
|
||||||
} catch (IOException e) {
|
|
||||||
|
// Punkte speichern
|
||||||
|
List<Map<String, Object>> data = new ArrayList<>();
|
||||||
|
for (int i = 0; i < points.size(); i++) {
|
||||||
|
Map<String, Object> entry = new LinkedHashMap<>();
|
||||||
|
entry.put("location", points.get(i));
|
||||||
|
entry.put("label", labels.get(i));
|
||||||
|
data.add(entry);
|
||||||
|
}
|
||||||
|
config.set("intro-points", data);
|
||||||
|
config.set("points", null);
|
||||||
|
|
||||||
|
try {
|
||||||
|
config.save(configFile);
|
||||||
|
} catch (IOException e) {
|
||||||
NexusLobby.getInstance().getLogger().severe("Fehler beim Speichern der Intro-Config: " + e.getMessage());
|
NexusLobby.getInstance().getLogger().severe("Fehler beim Speichern der Intro-Config: " + e.getMessage());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
// Events
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
@EventHandler
|
@EventHandler
|
||||||
public void onJoin(PlayerJoinEvent event) {
|
public void onJoin(PlayerJoinEvent event) {
|
||||||
if (!event.getPlayer().hasPlayedBefore() && points.size() >= 2) {
|
if (!event.getPlayer().hasPlayedBefore() && points.size() >= 2) {
|
||||||
@@ -84,12 +141,9 @@ public class IntroModule implements Module, Listener, CommandExecutor {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@EventHandler
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
public void onSneak(PlayerToggleSneakEvent event) {
|
// Befehle
|
||||||
if (activeIntro.contains(event.getPlayer().getUniqueId()) && event.isSneaking()) {
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
stopIntro(event.getPlayer(), true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean onCommand(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, String[] args) {
|
public boolean onCommand(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, String[] args) {
|
||||||
@@ -97,26 +151,52 @@ public class IntroModule implements Module, Listener, CommandExecutor {
|
|||||||
if (!p.hasPermission("nexuslobby.admin")) return true;
|
if (!p.hasPermission("nexuslobby.admin")) return true;
|
||||||
|
|
||||||
if (args.length == 0) {
|
if (args.length == 0) {
|
||||||
p.sendMessage("§8§m------------------------------------");
|
sendHelp(p);
|
||||||
p.sendMessage("§6§lNexus Intro System (Cinematic)");
|
|
||||||
p.sendMessage("§e/intro add §7- Punkt hinzufügen");
|
|
||||||
p.sendMessage("§e/intro clear §7- Alle Punkte löschen");
|
|
||||||
p.sendMessage("§e/intro start §7- Teste die Fahrt");
|
|
||||||
p.sendMessage("§8§m------------------------------------");
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (args[0].toLowerCase()) {
|
switch (args[0].toLowerCase()) {
|
||||||
|
|
||||||
case "add" -> {
|
case "add" -> {
|
||||||
|
String pointLabel = args.length > 1
|
||||||
|
? String.join(" ", Arrays.copyOfRange(args, 1, args.length))
|
||||||
|
: "";
|
||||||
points.add(p.getLocation());
|
points.add(p.getLocation());
|
||||||
savePoints();
|
labels.add(pointLabel);
|
||||||
p.sendMessage("§8[§6Nexus§8] §aPunkt #" + points.size() + " wurde gesetzt!");
|
saveConfig();
|
||||||
|
if (pointLabel.isEmpty()) {
|
||||||
|
p.sendMessage("§8[§6Nexus§8] §aPunkt §e#" + points.size() + " §awurde gesetzt.");
|
||||||
|
} else {
|
||||||
|
p.sendMessage("§8[§6Nexus§8] §aPunkt §e#" + points.size() + " §amit Text §e\"" + pointLabel + "\" §agesetzt.");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
case "clear" -> {
|
case "clear" -> {
|
||||||
points.clear();
|
points.clear();
|
||||||
savePoints();
|
labels.clear();
|
||||||
|
saveConfig();
|
||||||
p.sendMessage("§8[§6Nexus§8] §cAlle Intro-Punkte wurden gelöscht.");
|
p.sendMessage("§8[§6Nexus§8] §cAlle Intro-Punkte wurden gelöscht.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case "list" -> {
|
||||||
|
if (points.isEmpty()) {
|
||||||
|
p.sendMessage("§8[§6Nexus§8] §cKeine Punkte vorhanden.");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
p.sendMessage("§8§m------------------------------------");
|
||||||
|
p.sendMessage("§6§lIntro-Punkte (" + points.size() + ")");
|
||||||
|
for (int i = 0; i < points.size(); i++) {
|
||||||
|
Location loc = points.get(i);
|
||||||
|
String lbl = labels.get(i).isEmpty() ? "§7(kein Text)" : "§e" + labels.get(i);
|
||||||
|
p.sendMessage("§8#" + (i + 1) + " §7"
|
||||||
|
+ String.format("%.1f", loc.getX()) + ", "
|
||||||
|
+ String.format("%.1f", loc.getY()) + ", "
|
||||||
|
+ String.format("%.1f", loc.getZ())
|
||||||
|
+ " " + lbl);
|
||||||
|
}
|
||||||
|
p.sendMessage("§8§m------------------------------------");
|
||||||
|
}
|
||||||
|
|
||||||
case "start" -> {
|
case "start" -> {
|
||||||
if (points.size() < 2) {
|
if (points.size() < 2) {
|
||||||
p.sendMessage("§8[§6Nexus§8] §cDu brauchst mindestens 2 Punkte für eine Fahrt.");
|
p.sendMessage("§8[§6Nexus§8] §cDu brauchst mindestens 2 Punkte für eine Fahrt.");
|
||||||
@@ -124,18 +204,69 @@ public class IntroModule implements Module, Listener, CommandExecutor {
|
|||||||
startIntro(p);
|
startIntro(p);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// /intro setendtitle <Text> — setzt den großen Abschluss-Titel
|
||||||
|
case "setendtitle" -> {
|
||||||
|
if (args.length < 2) {
|
||||||
|
p.sendMessage("§8[§6Nexus§8] §cBenutzung: §e/intro setendtitle <Text>");
|
||||||
|
p.sendMessage("§7Farbcodes mit §e& §7möglich. Beispiel: §e/intro setendtitle &a&lWillkommen!");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
endTitle = colorize(String.join(" ", Arrays.copyOfRange(args, 1, args.length)));
|
||||||
|
saveConfig();
|
||||||
|
p.sendMessage("§8[§6Nexus§8] §aAbschluss-Titel gesetzt: " + endTitle);
|
||||||
|
// Vorschau
|
||||||
|
p.sendTitle(endTitle, endSubtitle, TITLE_FADE_IN, TITLE_STAY, TITLE_FADE_OUT);
|
||||||
|
}
|
||||||
|
|
||||||
|
// /intro setendsubtitle <Text> — setzt den kleinen Abschluss-Untertitel
|
||||||
|
case "setendsubtitle" -> {
|
||||||
|
if (args.length < 2) {
|
||||||
|
p.sendMessage("§8[§6Nexus§8] §cBenutzung: §e/intro setendsubtitle <Text>");
|
||||||
|
p.sendMessage("§7Farbcodes mit §e& §7möglich. Beispiel: §e/intro setendsubtitle &7Viel Spaß!");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
endSubtitle = colorize(String.join(" ", Arrays.copyOfRange(args, 1, args.length)));
|
||||||
|
saveConfig();
|
||||||
|
p.sendMessage("§8[§6Nexus§8] §aAbschluss-Untertitel gesetzt: " + endSubtitle);
|
||||||
|
// Vorschau
|
||||||
|
p.sendTitle(endTitle, endSubtitle, TITLE_FADE_IN, TITLE_STAY, TITLE_FADE_OUT);
|
||||||
|
}
|
||||||
|
|
||||||
|
default -> sendHelp(p);
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void sendHelp(Player p) {
|
||||||
|
p.sendMessage("§8§m------------------------------------");
|
||||||
|
p.sendMessage("§6§lNexus Intro System (Cinematic)");
|
||||||
|
p.sendMessage("§e/intro add [Text] §7- Punkt + optionaler Titel");
|
||||||
|
p.sendMessage("§e/intro clear §7- Alle Punkte löschen");
|
||||||
|
p.sendMessage("§e/intro list §7- Alle Punkte anzeigen");
|
||||||
|
p.sendMessage("§e/intro start §7- Fahrt testen");
|
||||||
|
p.sendMessage("§e/intro setendtitle <Text> §7- Abschluss-Titel setzen");
|
||||||
|
p.sendMessage("§e/intro setendsubtitle <Text> §7- Abschluss-Untertitel setzen");
|
||||||
|
p.sendMessage("§7Aktuell: §r" + endTitle + " §8/ §r" + endSubtitle);
|
||||||
|
p.sendMessage("§8§m------------------------------------");
|
||||||
|
}
|
||||||
|
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
// Intro-Logik
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
public void startIntro(Player player) {
|
public void startIntro(Player player) {
|
||||||
activeIntro.add(player.getUniqueId());
|
activeIntro.add(player.getUniqueId());
|
||||||
player.setGameMode(GameMode.SPECTATOR);
|
player.setGameMode(GameMode.SPECTATOR);
|
||||||
|
|
||||||
|
// Titel für den ersten Punkt sofort anzeigen
|
||||||
|
showTitleForSegment(player, 0);
|
||||||
|
|
||||||
new BukkitRunnable() {
|
new BukkitRunnable() {
|
||||||
int currentSegment = 0;
|
int currentSegment = 0;
|
||||||
int tickInSegment = 0;
|
int tickInSegment = 0;
|
||||||
boolean isPausing = true;
|
boolean isPausing = true;
|
||||||
|
int lastTitleSegment = 0;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void run() {
|
public void run() {
|
||||||
@@ -146,79 +277,89 @@ public class IntroModule implements Module, Listener, CommandExecutor {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (currentSegment >= points.size() - 1) {
|
if (currentSegment >= points.size() - 1) {
|
||||||
stopIntro(player, false);
|
stopIntro(player);
|
||||||
this.cancel();
|
this.cancel();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
Location start = points.get(currentSegment);
|
Location start = points.get(currentSegment);
|
||||||
Location end = points.get(currentSegment + 1);
|
Location end = points.get(currentSegment + 1);
|
||||||
|
|
||||||
if (isPausing) {
|
if (isPausing) {
|
||||||
// Kamera steht am aktuellen Punkt
|
|
||||||
player.teleport(start);
|
player.teleport(start);
|
||||||
tickInSegment++;
|
tickInSegment++;
|
||||||
if (tickInSegment >= TICKS_PAUSE) {
|
if (tickInSegment >= TICKS_PAUSE) {
|
||||||
isPausing = false;
|
isPausing = false;
|
||||||
tickInSegment = 0;
|
tickInSegment = 0;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Kamera fliegt zum nächsten Punkt
|
|
||||||
double progress = (double) tickInSegment / (double) TICKS_FLUG;
|
double progress = (double) tickInSegment / (double) TICKS_FLUG;
|
||||||
|
double smoothT = progress * progress * (3 - 2 * progress);
|
||||||
// "Smooth Step" für flüssigeres Beschleunigen/Bremsen
|
|
||||||
double smoothT = progress * progress * (3 - 2 * progress);
|
player.teleport(interpolate(start, end, smoothT));
|
||||||
|
|
||||||
Location nextLoc = interpolate(start, end, smoothT);
|
|
||||||
player.teleport(nextLoc);
|
|
||||||
|
|
||||||
tickInSegment++;
|
tickInSegment++;
|
||||||
if (tickInSegment >= TICKS_FLUG) {
|
if (tickInSegment >= TICKS_FLUG) {
|
||||||
isPausing = true;
|
isPausing = true;
|
||||||
tickInSegment = 0;
|
tickInSegment = 0;
|
||||||
currentSegment++;
|
currentSegment++;
|
||||||
|
|
||||||
|
if (currentSegment < points.size() && currentSegment != lastTitleSegment) {
|
||||||
|
showTitleForSegment(player, currentSegment);
|
||||||
|
lastTitleSegment = currentSegment;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
player.spigot().sendMessage(ChatMessageType.ACTION_BAR,
|
|
||||||
new TextComponent("§6§lINTRO-TOUR §8| §ePunkt " + (currentSegment + 1) + " §8| §7Sneak zum Abbrechen"));
|
|
||||||
|
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
this.cancel();
|
this.cancel();
|
||||||
stopIntro(player, true);
|
stopIntro(player);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}.runTaskTimer(NexusLobby.getInstance(), 0L, 1L);
|
}.runTaskTimer(NexusLobby.getInstance(), 0L, 1L);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void showTitleForSegment(Player player, int segmentIndex) {
|
||||||
|
if (segmentIndex < 0 || segmentIndex >= labels.size()) return;
|
||||||
|
String lbl = labels.get(segmentIndex);
|
||||||
|
if (lbl == null || lbl.isEmpty()) return;
|
||||||
|
|
||||||
|
player.sendTitle(
|
||||||
|
"§6§l" + lbl,
|
||||||
|
"§7✦ NexusLobby ✦",
|
||||||
|
TITLE_FADE_IN,
|
||||||
|
TITLE_STAY,
|
||||||
|
TITLE_FADE_OUT
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
private Location interpolate(Location start, Location end, double t) {
|
private Location interpolate(Location start, Location end, double t) {
|
||||||
double x = start.getX() + (end.getX() - start.getX()) * t;
|
double x = start.getX() + (end.getX() - start.getX()) * t;
|
||||||
double y = start.getY() + (end.getY() - start.getY()) * t;
|
double y = start.getY() + (end.getY() - start.getY()) * t;
|
||||||
double z = start.getZ() + (end.getZ() - start.getZ()) * t;
|
double z = start.getZ() + (end.getZ() - start.getZ()) * t;
|
||||||
|
|
||||||
// Sanfte Drehung
|
|
||||||
float startYaw = start.getYaw();
|
|
||||||
float endYaw = end.getYaw();
|
|
||||||
// Verhindert ruckartige 360-Grad Dreher
|
|
||||||
float diff = (endYaw - startYaw) % 360;
|
|
||||||
if (diff > 180) diff -= 360;
|
|
||||||
if (diff < -180) diff += 360;
|
|
||||||
float yaw = startYaw + diff * (float)t;
|
|
||||||
|
|
||||||
|
float startYaw = start.getYaw();
|
||||||
|
float endYaw = end.getYaw();
|
||||||
|
float diff = (endYaw - startYaw) % 360;
|
||||||
|
if (diff > 180) diff -= 360;
|
||||||
|
if (diff < -180) diff += 360;
|
||||||
|
float yaw = startYaw + diff * (float) t;
|
||||||
float pitch = (float) (start.getPitch() + (end.getPitch() - start.getPitch()) * t);
|
float pitch = (float) (start.getPitch() + (end.getPitch() - start.getPitch()) * t);
|
||||||
|
|
||||||
return new Location(start.getWorld(), x, y, z, yaw, pitch);
|
return new Location(start.getWorld(), x, y, z, yaw, pitch);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void stopIntro(Player player, boolean canceled) {
|
private void stopIntro(Player player) {
|
||||||
activeIntro.remove(player.getUniqueId());
|
activeIntro.remove(player.getUniqueId());
|
||||||
player.setGameMode(GameMode.ADVENTURE);
|
player.setGameMode(GameMode.ADVENTURE);
|
||||||
player.teleport(player.getWorld().getSpawnLocation());
|
player.teleport(player.getWorld().getSpawnLocation());
|
||||||
|
|
||||||
if (canceled) {
|
// Abschluss-Titel — komplett konfigurierbar
|
||||||
player.sendMessage("§8[§6Nexus§8] §cIntro abgebrochen.");
|
player.sendTitle(endTitle, endSubtitle, 10, 60, 20);
|
||||||
} else {
|
}
|
||||||
player.sendMessage("§8[§6Nexus§8] §aWillkommen auf dem Netzwerk!");
|
|
||||||
}
|
/** Ersetzt &-Farbcodes durch §-Codes */
|
||||||
|
private String colorize(String input) {
|
||||||
|
return input.replace("&", "§");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -18,48 +18,43 @@ import java.util.UUID;
|
|||||||
public class ParkourListener implements Listener {
|
public class ParkourListener implements Listener {
|
||||||
|
|
||||||
private final ParkourManager manager;
|
private final ParkourManager manager;
|
||||||
private final String NPC_TAG = "parkour_npc";
|
private final String NPC_TAG = "parkour_npc";
|
||||||
private final HashMap<UUID, Long> startCooldown = new HashMap<>();
|
private final HashMap<UUID, Long> startCooldown = new HashMap<>();
|
||||||
|
|
||||||
public ParkourListener(ParkourManager manager) {
|
public ParkourListener(ParkourManager manager) {
|
||||||
this.manager = manager;
|
this.manager = manager;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Startet den Parkour per Rechtsklick auf den ArmorStand
|
|
||||||
*/
|
|
||||||
@EventHandler
|
@EventHandler
|
||||||
public void onInteract(PlayerInteractAtEntityEvent event) {
|
public void onInteract(PlayerInteractAtEntityEvent event) {
|
||||||
if (!(event.getRightClicked() instanceof ArmorStand as)) return;
|
if (!(event.getRightClicked() instanceof ArmorStand as)) return;
|
||||||
|
if (!as.getScoreboardTags().contains(NPC_TAG)) return;
|
||||||
// Prüfen, ob der ArmorStand den richtigen Tag hat
|
|
||||||
if (as.getScoreboardTags().contains(NPC_TAG)) {
|
|
||||||
Player player = event.getPlayer();
|
|
||||||
|
|
||||||
// Abbrechen, wenn der Spieler schon im Parkour ist
|
|
||||||
if (manager.isIngame(player)) return;
|
|
||||||
|
|
||||||
// Cooldown-Check (3 Sekunden)
|
Player player = event.getPlayer();
|
||||||
if (startCooldown.containsKey(player.getUniqueId())) {
|
event.setCancelled(true);
|
||||||
if (System.currentTimeMillis() - startCooldown.get(player.getUniqueId()) < 3000) {
|
|
||||||
player.sendMessage("§cBitte warte einen Moment, bevor du erneut startest.");
|
if (manager.isIngame(player)) return;
|
||||||
return;
|
|
||||||
}
|
// Cooldown-Check (3 Sekunden)
|
||||||
|
if (startCooldown.containsKey(player.getUniqueId())) {
|
||||||
|
if (System.currentTimeMillis() - startCooldown.get(player.getUniqueId()) < 3000) {
|
||||||
|
player.sendMessage("§cBitte warte einen Moment, bevor du erneut startest.");
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Parkour starten
|
|
||||||
manager.startParkour(player, as.getLocation());
|
|
||||||
player.sendMessage("§8[§6Parkour§8] §eViel Erfolg! Erreiche das Ziel so schnell wie möglich.");
|
|
||||||
player.playSound(player.getLocation(), Sound.ENTITY_EXPERIENCE_ORB_PICKUP, 1f, 1f);
|
|
||||||
|
|
||||||
// Event abbrechen, damit man keine Ausrüstung vom ArmorStand klaut
|
|
||||||
event.setCancelled(true);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
manager.startParkour(player, as.getLocation());
|
||||||
|
// Strecken-Nummer dem Spieler mitteilen
|
||||||
|
int track = manager.getActiveTrack(player);
|
||||||
|
if (track != -1) {
|
||||||
|
player.sendMessage("§8[§6Parkour§8] §eViel Erfolg auf §bStrecke " + track + "§e! Erreiche das Ziel so schnell wie möglich.");
|
||||||
|
}
|
||||||
|
player.playSound(player.getLocation(), Sound.ENTITY_EXPERIENCE_ORB_PICKUP, 1f, 1f);
|
||||||
}
|
}
|
||||||
|
|
||||||
@EventHandler
|
@EventHandler
|
||||||
public void onMove(PlayerMoveEvent event) {
|
public void onMove(PlayerMoveEvent event) {
|
||||||
// Performance: Nur berechnen, wenn ein voller Block gewechselt wurde
|
// Performance: Nur bei Block-Wechsel berechnen
|
||||||
if (event.getFrom().getBlockX() == event.getTo().getBlockX() &&
|
if (event.getFrom().getBlockX() == event.getTo().getBlockX() &&
|
||||||
event.getFrom().getBlockY() == event.getTo().getBlockY() &&
|
event.getFrom().getBlockY() == event.getTo().getBlockY() &&
|
||||||
event.getFrom().getBlockZ() == event.getTo().getBlockZ()) return;
|
event.getFrom().getBlockZ() == event.getTo().getBlockZ()) return;
|
||||||
@@ -70,8 +65,9 @@ public class ParkourListener implements Listener {
|
|||||||
Location loc = player.getLocation();
|
Location loc = player.getLocation();
|
||||||
|
|
||||||
// --- 1. ABSTURZ-CHECK ---
|
// --- 1. ABSTURZ-CHECK ---
|
||||||
// Passe die Höhe '50' an deine Map an!
|
// Absturz-Y aus config lesen (Standard: 50), damit Maps unter Y=50 korrekt funktionieren
|
||||||
if (loc.getY() < 50) {
|
double fallY = NexusLobby.getInstance().getConfig().getDouble("parkour.fall_y", 50.0);
|
||||||
|
if (loc.getY() < fallY) {
|
||||||
Location lastCp = manager.getCheckpoint(player);
|
Location lastCp = manager.getCheckpoint(player);
|
||||||
if (lastCp != null) {
|
if (lastCp != null) {
|
||||||
player.teleport(lastCp);
|
player.teleport(lastCp);
|
||||||
@@ -81,27 +77,25 @@ public class ParkourListener implements Listener {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// --- 2. CHECKPOINT-CHECK ---
|
// --- 2. CHECKPOINT-CHECK (Track-spezifisch) ---
|
||||||
List<Location> cps = manager.getOrderedCheckpoints();
|
List<Location> cps = manager.getOrderedCheckpoints(player);
|
||||||
for (int i = 0; i < cps.size(); i++) {
|
for (int i = 0; i < cps.size(); i++) {
|
||||||
if (isNearby(loc, cps.get(i))) {
|
if (isNearby(loc, cps.get(i))) {
|
||||||
manager.reachCheckpoint(player, i);
|
manager.reachCheckpoint(player, i);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// --- 3. ZIEL-CHECK ---
|
// --- 3. ZIEL-CHECK (Track-spezifisch) ---
|
||||||
Location finish = manager.getFinishLocation();
|
Location finish = manager.getFinishLocation(player);
|
||||||
if (finish != null && isNearby(loc, finish)) {
|
if (finish != null && isNearby(loc, finish)) {
|
||||||
manager.finishParkour(player);
|
manager.finishParkour(player);
|
||||||
// Nach dem Ziel setzen wir einen Cooldown
|
|
||||||
startCooldown.put(player.getUniqueId(), System.currentTimeMillis());
|
startCooldown.put(player.getUniqueId(), System.currentTimeMillis());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean isNearby(Location playerLoc, Location targetLoc) {
|
private boolean isNearby(Location playerLoc, Location targetLoc) {
|
||||||
if (playerLoc.getWorld() == null || !playerLoc.getWorld().equals(targetLoc.getWorld())) return false;
|
if (playerLoc.getWorld() == null || !playerLoc.getWorld().equals(targetLoc.getWorld())) return false;
|
||||||
// 2.25 entspricht einem Radius von ca. 1.5 Blöcken
|
return playerLoc.distanceSquared(targetLoc) <= 2.25; // ~1.5 Blöcke Radius
|
||||||
return playerLoc.distanceSquared(targetLoc) <= 2.25;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@EventHandler
|
@EventHandler
|
||||||
|
|||||||
@@ -17,71 +17,225 @@ import java.io.IOException;
|
|||||||
import java.util.*;
|
import java.util.*;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ParkourManager – unterstützt zwei Strecken (Track 1 & 2).
|
||||||
|
*
|
||||||
|
* Config-Struktur (parkour.yml):
|
||||||
|
* tracks:
|
||||||
|
* 1:
|
||||||
|
* start: <Location>
|
||||||
|
* finish: <Location>
|
||||||
|
* checkpoints:
|
||||||
|
* 1: <Location>
|
||||||
|
* 2: <Location>
|
||||||
|
* ...
|
||||||
|
* 2:
|
||||||
|
* start: <Location>
|
||||||
|
* finish: <Location>
|
||||||
|
* checkpoints:
|
||||||
|
* 1: <Location>
|
||||||
|
* ...
|
||||||
|
* besttimes:
|
||||||
|
* <uuid>: <seconds>
|
||||||
|
* names:
|
||||||
|
* <uuid>: <playerName>
|
||||||
|
*
|
||||||
|
* Beim Start wird zufällig eine der beiden konfigurierten Strecken ausgewählt.
|
||||||
|
* Beide Strecken müssen die gleiche Anzahl an Checkpoints haben – andernfalls
|
||||||
|
* wird nur die vollständige gewählt.
|
||||||
|
*/
|
||||||
public class ParkourManager {
|
public class ParkourManager {
|
||||||
|
|
||||||
private final NexusLobby plugin;
|
private final NexusLobby plugin;
|
||||||
private final File file;
|
private final File file;
|
||||||
private FileConfiguration config;
|
private FileConfiguration config;
|
||||||
|
|
||||||
private final Map<UUID, Long> startTime = new HashMap<>();
|
// Aktive Läufe
|
||||||
private final Map<UUID, Location> lastCheckpointLoc = new HashMap<>();
|
private final Map<UUID, Long> startTime = new HashMap<>();
|
||||||
private final Map<UUID, Integer> nextCheckpointIndex = new HashMap<>();
|
private final Map<UUID, Location> lastCheckpointLoc = new HashMap<>();
|
||||||
|
private final Map<UUID, Integer> nextCheckpointIndex= new HashMap<>();
|
||||||
|
/** Welche Track-Nummer (1 oder 2) der Spieler gerade läuft */
|
||||||
|
private final Map<UUID, Integer> activeTrack = new HashMap<>();
|
||||||
|
|
||||||
|
private final Random random = new Random();
|
||||||
|
|
||||||
public ParkourManager(NexusLobby plugin) {
|
public ParkourManager(NexusLobby plugin) {
|
||||||
this.plugin = plugin;
|
this.plugin = plugin;
|
||||||
this.file = new File(plugin.getDataFolder(), "parkour.yml");
|
this.file = new File(plugin.getDataFolder(), "parkour.yml");
|
||||||
loadConfig();
|
loadConfig();
|
||||||
|
migrateLegacyBestTimes();
|
||||||
startParticleTask();
|
startParticleTask();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
// Config I/O
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
private void loadConfig() {
|
private void loadConfig() {
|
||||||
if (!file.exists()) {
|
if (!file.exists()) {
|
||||||
file.getParentFile().mkdirs();
|
file.getParentFile().mkdirs();
|
||||||
try {
|
try {
|
||||||
file.createNewFile();
|
file.createNewFile();
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
plugin.getLogger().severe("Fehler beim Erstellen der Parkour-Datei: " + e.getMessage());
|
plugin.getLogger().severe("Fehler beim Erstellen der Parkour-Datei: " + e.getMessage());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
config = YamlConfiguration.loadConfiguration(file);
|
config = YamlConfiguration.loadConfiguration(file);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setCheckpoint(Player player, Location loc) {
|
/**
|
||||||
int nextId = 1;
|
* Migriert Bestzeiten vom alten Format (besttimes.<uuid>) ins neue Format
|
||||||
if (config.contains("locations.checkpoints") && config.getConfigurationSection("locations.checkpoints") != null) {
|
* (besttimes.<track>.<uuid>). Alte Zeiten werden Track 1 zugeordnet, da vor
|
||||||
nextId = config.getConfigurationSection("locations.checkpoints").getKeys(false).size() + 1;
|
* dieser Änderung nur ein Track existierte.
|
||||||
|
* Läuft einmalig beim Start; tut nichts wenn keine alten Daten vorhanden sind.
|
||||||
|
*/
|
||||||
|
private void migrateLegacyBestTimes() {
|
||||||
|
if (!config.contains("besttimes")) return;
|
||||||
|
|
||||||
|
// Prüfen ob es direkte UUID-Keys unter besttimes gibt (altes Format)
|
||||||
|
// Im neuen Format wären hier nur "1" und "2" als Keys
|
||||||
|
var section = config.getConfigurationSection("besttimes");
|
||||||
|
if (section == null) return;
|
||||||
|
|
||||||
|
int migrated = 0;
|
||||||
|
for (String key : new java.util.HashSet<>(section.getKeys(false))) {
|
||||||
|
// Neue Keys sind "1" oder "2" – alte Keys sind UUID-Strings (36 Zeichen, enthalten "-")
|
||||||
|
if (!key.equals("1") && !key.equals("2") && key.contains("-")) {
|
||||||
|
double time = config.getDouble("besttimes." + key);
|
||||||
|
// Nur migrieren wenn unter besttimes.1.<uuid> noch kein besserer Wert steht
|
||||||
|
String newPath = "besttimes.1." + key;
|
||||||
|
double existingTime = config.getDouble(newPath, 99999.9);
|
||||||
|
if (time < existingTime) {
|
||||||
|
config.set(newPath, time);
|
||||||
|
}
|
||||||
|
// Alten Key entfernen
|
||||||
|
config.set("besttimes." + key, null);
|
||||||
|
migrated++;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
addCheckpointLocation(String.valueOf(nextId), loc);
|
if (migrated > 0) {
|
||||||
player.sendMessage("§8[§6Parkour§8] §7Checkpoint §e#" + nextId + " §7an deiner Position gesetzt.");
|
save();
|
||||||
|
plugin.getLogger().info("[Parkour] Migration: " + migrated
|
||||||
|
+ " Bestzeit(en) von altem Format nach Strecke 1 übertragen.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void save() {
|
||||||
|
try {
|
||||||
|
config.save(file);
|
||||||
|
} catch (IOException e) {
|
||||||
|
plugin.getLogger().severe("Fehler beim Speichern der Parkour-Config: " + e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
// Setup-Befehle (Admin)
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
public void setStartLocation(Player player, int track) {
|
||||||
|
config.set("tracks." + track + ".start", player.getLocation());
|
||||||
|
save();
|
||||||
|
player.sendMessage("§8[§6Parkour§8] §7Start von §eStrecke " + track + " §7an deiner Position gesetzt.");
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setFinishLocation(Player player, int track) {
|
||||||
|
config.set("tracks." + track + ".finish", player.getLocation());
|
||||||
|
save();
|
||||||
|
player.sendMessage("§8[§6Parkour§8] §7Ziel von §eStrecke " + track + " §7an deiner Position gesetzt.");
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setCheckpoint(Player player, int track) {
|
||||||
|
String cpBase = "tracks." + track + ".checkpoints";
|
||||||
|
int nextId = 1;
|
||||||
|
if (config.contains(cpBase) && config.getConfigurationSection(cpBase) != null) {
|
||||||
|
nextId = config.getConfigurationSection(cpBase).getKeys(false).size() + 1;
|
||||||
|
}
|
||||||
|
config.set(cpBase + "." + nextId, player.getLocation());
|
||||||
|
save();
|
||||||
|
player.sendMessage("§8[§6Parkour§8] §7Checkpoint §e#" + nextId + " §7(Strecke " + track + ") §7gesetzt.");
|
||||||
player.playSound(player.getLocation(), Sound.BLOCK_AMETHYST_BLOCK_CHIME, 1.0f, 1.5f);
|
player.playSound(player.getLocation(), Sound.BLOCK_AMETHYST_BLOCK_CHIME, 1.0f, 1.5f);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Löscht die gesamte Strecke (Checkpoints und Ziel) und bricht aktuelle Läufe ab.
|
* Löscht alle Punkte beider Strecken und bricht laufende Runs ab.
|
||||||
*/
|
*/
|
||||||
public void removeAllPoints() {
|
public void removeAllPoints() {
|
||||||
config.set("locations.checkpoints", null);
|
config.set("tracks", null);
|
||||||
config.set("locations.finish", null);
|
|
||||||
save();
|
save();
|
||||||
|
|
||||||
// Alle Spieler aus dem Parkour werfen, damit keine Partikel zu gelöschten Zielen führen
|
|
||||||
for (UUID uuid : new HashSet<>(startTime.keySet())) {
|
for (UUID uuid : new HashSet<>(startTime.keySet())) {
|
||||||
Player p = Bukkit.getPlayer(uuid);
|
Player p = Bukkit.getPlayer(uuid);
|
||||||
if (p != null) {
|
if (p != null) {
|
||||||
stopParkour(p);
|
stopParkour(p);
|
||||||
p.sendMessage("§8[§6Parkour§8] §cDie aktuelle Strecke wurde soeben gelöscht.");
|
p.sendMessage("§8[§6Parkour§8] §cDie Strecken wurden soeben gelöscht.");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void startParkour(Player player, Location startLoc) {
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
// Getter (Track-spezifisch)
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
public Location getStartLocation(int track) {
|
||||||
|
return config.getLocation("tracks." + track + ".start");
|
||||||
|
}
|
||||||
|
|
||||||
|
public Location getFinishLocation(int track) {
|
||||||
|
return config.getLocation("tracks." + track + ".finish");
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<Location> getOrderedCheckpoints(int track) {
|
||||||
|
String cpBase = "tracks." + track + ".checkpoints";
|
||||||
|
if (!config.contains(cpBase) || config.getConfigurationSection(cpBase) == null)
|
||||||
|
return new ArrayList<>();
|
||||||
|
|
||||||
|
return config.getConfigurationSection(cpBase).getKeys(false).stream()
|
||||||
|
.sorted(Comparator.comparingInt(Integer::parseInt))
|
||||||
|
.map(key -> config.getLocation(cpBase + "." + key))
|
||||||
|
.filter(Objects::nonNull)
|
||||||
|
.collect(Collectors.toList());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gibt die aktive Track-Nummer des Spielers zurück (1 oder 2), oder -1 wenn nicht aktiv.
|
||||||
|
*/
|
||||||
|
public int getActiveTrack(Player player) {
|
||||||
|
return activeTrack.getOrDefault(player.getUniqueId(), -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Convenience-Wrapper für ParkourListener (nutzt den aktiven Track des Spielers)
|
||||||
|
public List<Location> getOrderedCheckpoints(Player player) {
|
||||||
|
return getOrderedCheckpoints(getActiveTrack(player));
|
||||||
|
}
|
||||||
|
|
||||||
|
public Location getFinishLocation(Player player) {
|
||||||
|
return getFinishLocation(getActiveTrack(player));
|
||||||
|
}
|
||||||
|
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
// Parkour-Ablauf
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Startet den Parkour für den Spieler.
|
||||||
|
* Es wird zufällig eine der beiden Strecken gewählt, sofern beide vollständig
|
||||||
|
* konfiguriert sind (Start + Finish + mindestens gleiche Checkpoint-Anzahl).
|
||||||
|
* Falls nur eine Strecke konfiguriert ist, wird diese gewählt.
|
||||||
|
*/
|
||||||
|
public void startParkour(Player player, Location npcLocation) {
|
||||||
if (startTime.containsKey(player.getUniqueId())) return;
|
if (startTime.containsKey(player.getUniqueId())) return;
|
||||||
|
|
||||||
|
int chosenTrack = pickRandomTrack();
|
||||||
|
if (chosenTrack == -1) {
|
||||||
|
player.sendMessage("§8[§6Parkour§8] §cKeine Strecke konfiguriert. Bitte einen Admin kontaktieren.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
startTime.put(player.getUniqueId(), System.currentTimeMillis());
|
startTime.put(player.getUniqueId(), System.currentTimeMillis());
|
||||||
lastCheckpointLoc.put(player.getUniqueId(), startLoc);
|
lastCheckpointLoc.put(player.getUniqueId(), npcLocation);
|
||||||
nextCheckpointIndex.put(player.getUniqueId(), 0);
|
nextCheckpointIndex.put(player.getUniqueId(), 0);
|
||||||
|
activeTrack.put(player.getUniqueId(), chosenTrack);
|
||||||
|
|
||||||
player.playSound(player.getLocation(), Sound.BLOCK_NOTE_BLOCK_CHIME, 1.0f, 1.2f);
|
player.playSound(player.getLocation(), Sound.BLOCK_NOTE_BLOCK_CHIME, 1.0f, 1.2f);
|
||||||
|
|
||||||
new BukkitRunnable() {
|
new BukkitRunnable() {
|
||||||
@@ -93,12 +247,45 @@ public class ParkourManager {
|
|||||||
}
|
}
|
||||||
long diff = System.currentTimeMillis() - startTime.get(player.getUniqueId());
|
long diff = System.currentTimeMillis() - startTime.get(player.getUniqueId());
|
||||||
double sec = diff / 1000.0;
|
double sec = diff / 1000.0;
|
||||||
player.spigot().sendMessage(ChatMessageType.ACTION_BAR,
|
player.spigot().sendMessage(ChatMessageType.ACTION_BAR,
|
||||||
new TextComponent("§6⏱ Zeit: §e" + String.format("%.2f", sec) + "s §8| §bNächster Punkt: §7Partikel folgen"));
|
new TextComponent("§6⏱ Zeit: §e" + String.format("%.2f", sec) + "s §8| §bNächster Punkt: §7Partikel folgen"));
|
||||||
}
|
}
|
||||||
}.runTaskTimer(plugin, 0L, 1L);
|
}.runTaskTimer(plugin, 0L, 1L);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Wählt zufällig einen validen Track (1 oder 2).
|
||||||
|
* Bevorzugt Tracks mit gleichem Checkpoint-Count; fällt auf jeden konfigurierten zurück.
|
||||||
|
* Gibt -1 zurück, wenn kein Track konfiguriert ist.
|
||||||
|
*/
|
||||||
|
private int pickRandomTrack() {
|
||||||
|
boolean t1 = isTrackReady(1);
|
||||||
|
boolean t2 = isTrackReady(2);
|
||||||
|
|
||||||
|
if (t1 && t2) {
|
||||||
|
// Beide vollständig – prüfen ob gleiche Länge
|
||||||
|
int len1 = getOrderedCheckpoints(1).size();
|
||||||
|
int len2 = getOrderedCheckpoints(2).size();
|
||||||
|
if (len1 != len2) {
|
||||||
|
// Warnung ins Log, aber trotzdem zufällig wählen
|
||||||
|
plugin.getLogger().warning("[Parkour] Strecke 1 hat " + len1 + " Checkpoints, Strecke 2 hat " + len2 + ". Bitte angleichen!");
|
||||||
|
}
|
||||||
|
return random.nextBoolean() ? 1 : 2;
|
||||||
|
} else if (t1) {
|
||||||
|
return 1;
|
||||||
|
} else if (t2) {
|
||||||
|
return 2;
|
||||||
|
}
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Gibt zurück, ob Start, Finish und mindestens 1 Checkpoint für den Track gesetzt sind. */
|
||||||
|
public boolean isTrackReady(int track) {
|
||||||
|
return getStartLocation(track) != null
|
||||||
|
&& getFinishLocation(track) != null
|
||||||
|
&& !getOrderedCheckpoints(track).isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
private void startParticleTask() {
|
private void startParticleTask() {
|
||||||
new BukkitRunnable() {
|
new BukkitRunnable() {
|
||||||
@Override
|
@Override
|
||||||
@@ -107,8 +294,9 @@ public class ParkourManager {
|
|||||||
Player p = Bukkit.getPlayer(uuid);
|
Player p = Bukkit.getPlayer(uuid);
|
||||||
if (p == null) continue;
|
if (p == null) continue;
|
||||||
|
|
||||||
|
int track = activeTrack.getOrDefault(uuid, 1);
|
||||||
int nextIdx = nextCheckpointIndex.getOrDefault(uuid, 0);
|
int nextIdx = nextCheckpointIndex.getOrDefault(uuid, 0);
|
||||||
List<Location> cps = getOrderedCheckpoints();
|
List<Location> cps = getOrderedCheckpoints(track);
|
||||||
|
|
||||||
if (nextIdx < cps.size()) {
|
if (nextIdx < cps.size()) {
|
||||||
Location nextCp = cps.get(nextIdx);
|
Location nextCp = cps.get(nextIdx);
|
||||||
@@ -116,7 +304,7 @@ public class ParkourManager {
|
|||||||
p.spawnParticle(Particle.SOUL_FIRE_FLAME, nextCp.clone().add(0, 0.5, 0), 5, 0.1, 0.3, 0.1, 0.02);
|
p.spawnParticle(Particle.SOUL_FIRE_FLAME, nextCp.clone().add(0, 0.5, 0), 5, 0.1, 0.3, 0.1, 0.02);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
Location finish = getFinishLocation();
|
Location finish = getFinishLocation(track);
|
||||||
if (finish != null && p.getWorld().equals(finish.getWorld())) {
|
if (finish != null && p.getWorld().equals(finish.getWorld())) {
|
||||||
p.spawnParticle(Particle.HAPPY_VILLAGER, finish.clone().add(0, 0.5, 0), 8, 0.2, 0.5, 0.2, 0.02);
|
p.spawnParticle(Particle.HAPPY_VILLAGER, finish.clone().add(0, 0.5, 0), 8, 0.2, 0.5, 0.2, 0.02);
|
||||||
}
|
}
|
||||||
@@ -128,23 +316,24 @@ public class ParkourManager {
|
|||||||
|
|
||||||
public void reachCheckpoint(Player player, int reachedIndex) {
|
public void reachCheckpoint(Player player, int reachedIndex) {
|
||||||
int currentNext = nextCheckpointIndex.getOrDefault(player.getUniqueId(), 0);
|
int currentNext = nextCheckpointIndex.getOrDefault(player.getUniqueId(), 0);
|
||||||
|
if (reachedIndex != currentNext) return;
|
||||||
if (reachedIndex == currentNext) {
|
|
||||||
List<Location> cps = getOrderedCheckpoints();
|
int track = activeTrack.getOrDefault(player.getUniqueId(), 1);
|
||||||
if (reachedIndex < cps.size()) {
|
List<Location> cps = getOrderedCheckpoints(track);
|
||||||
lastCheckpointLoc.put(player.getUniqueId(), cps.get(reachedIndex));
|
if (reachedIndex >= cps.size()) return;
|
||||||
nextCheckpointIndex.put(player.getUniqueId(), reachedIndex + 1);
|
|
||||||
|
lastCheckpointLoc.put(player.getUniqueId(), cps.get(reachedIndex));
|
||||||
player.sendMessage("§8[§6Parkour§8] §bCheckpoint #" + (reachedIndex + 1) + " erreicht!");
|
nextCheckpointIndex.put(player.getUniqueId(), reachedIndex + 1);
|
||||||
player.playSound(player.getLocation(), Sound.ENTITY_EXPERIENCE_ORB_PICKUP, 0.8f, 1.5f);
|
|
||||||
}
|
player.sendMessage("§8[§6Parkour§8] §bCheckpoint #" + (reachedIndex + 1) + " erreicht!");
|
||||||
}
|
player.playSound(player.getLocation(), Sound.ENTITY_EXPERIENCE_ORB_PICKUP, 0.8f, 1.5f);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void finishParkour(Player player) {
|
public void finishParkour(Player player) {
|
||||||
if (!startTime.containsKey(player.getUniqueId())) return;
|
if (!startTime.containsKey(player.getUniqueId())) return;
|
||||||
|
|
||||||
if (nextCheckpointIndex.getOrDefault(player.getUniqueId(), 0) < getOrderedCheckpoints().size()) {
|
int track = activeTrack.getOrDefault(player.getUniqueId(), 1);
|
||||||
|
if (nextCheckpointIndex.getOrDefault(player.getUniqueId(), 0) < getOrderedCheckpoints(track).size()) {
|
||||||
player.sendMessage("§cDu hast Checkpoints übersprungen! Folge den Partikeln.");
|
player.sendMessage("§cDu hast Checkpoints übersprungen! Folge den Partikeln.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -153,12 +342,12 @@ public class ParkourManager {
|
|||||||
double seconds = duration / 1000.0;
|
double seconds = duration / 1000.0;
|
||||||
|
|
||||||
player.sendMessage("§8§m--------------------------------------");
|
player.sendMessage("§8§m--------------------------------------");
|
||||||
player.sendMessage("§8[§6Parkour§8] §a§lZiel erreicht!");
|
player.sendMessage("§8[§6Parkour§8] §a§lZiel erreicht! §8(Strecke " + track + ")");
|
||||||
player.sendMessage("§7Deine Zeit: §e" + String.format("%.2f", seconds) + "s");
|
player.sendMessage("§7Deine Zeit: §e" + String.format("%.2f", seconds) + "s");
|
||||||
player.sendMessage("§8§m--------------------------------------");
|
player.sendMessage("§8§m--------------------------------------");
|
||||||
|
|
||||||
player.playSound(player.getLocation(), Sound.ENTITY_PLAYER_LEVELUP, 1.0f, 1.0f);
|
player.playSound(player.getLocation(), Sound.ENTITY_PLAYER_LEVELUP, 1.0f, 1.0f);
|
||||||
|
|
||||||
saveBestTime(player, seconds);
|
saveBestTime(player, seconds);
|
||||||
stopParkour(player);
|
stopParkour(player);
|
||||||
}
|
}
|
||||||
@@ -167,45 +356,16 @@ public class ParkourManager {
|
|||||||
startTime.remove(player.getUniqueId());
|
startTime.remove(player.getUniqueId());
|
||||||
lastCheckpointLoc.remove(player.getUniqueId());
|
lastCheckpointLoc.remove(player.getUniqueId());
|
||||||
nextCheckpointIndex.remove(player.getUniqueId());
|
nextCheckpointIndex.remove(player.getUniqueId());
|
||||||
|
activeTrack.remove(player.getUniqueId());
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setStartLocation(Location loc) { config.set("locations.start", loc); save(); }
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
public void setFinishLocation(Location loc) { config.set("locations.finish", loc); save(); }
|
// Bestzeiten
|
||||||
public void addCheckpointLocation(String id, Location loc) { config.set("locations.checkpoints." + id, loc); save(); }
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
public Location getStartLocation() { return config.getLocation("locations.start"); }
|
|
||||||
public Location getFinishLocation() { return config.getLocation("locations.finish"); }
|
|
||||||
|
|
||||||
public List<Location> getOrderedCheckpoints() {
|
|
||||||
if (!config.contains("locations.checkpoints") || config.getConfigurationSection("locations.checkpoints") == null)
|
|
||||||
return new ArrayList<>();
|
|
||||||
|
|
||||||
return config.getConfigurationSection("locations.checkpoints").getKeys(false).stream()
|
|
||||||
.sorted(Comparator.comparingInt(Integer::parseInt))
|
|
||||||
.map(key -> config.getLocation("locations.checkpoints." + key))
|
|
||||||
.filter(Objects::nonNull)
|
|
||||||
.collect(Collectors.toList());
|
|
||||||
}
|
|
||||||
|
|
||||||
private void save() {
|
|
||||||
try {
|
|
||||||
config.save(file);
|
|
||||||
} catch (IOException e) {
|
|
||||||
plugin.getLogger().severe("Fehler beim Speichern der Parkour-Config: " + e.getMessage());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void clearStats() {
|
|
||||||
config.set("besttimes", null);
|
|
||||||
config.set("names", null);
|
|
||||||
save();
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean isIngame(Player player) { return startTime.containsKey(player.getUniqueId()); }
|
|
||||||
public Location getCheckpoint(Player player) { return lastCheckpointLoc.get(player.getUniqueId()); }
|
|
||||||
|
|
||||||
private void saveBestTime(Player player, double time) {
|
private void saveBestTime(Player player, double time) {
|
||||||
String path = "besttimes." + player.getUniqueId();
|
int track = activeTrack.getOrDefault(player.getUniqueId(), 1);
|
||||||
|
String path = "besttimes." + track + "." + player.getUniqueId();
|
||||||
double currentTime = config.getDouble(path, 99999.9);
|
double currentTime = config.getDouble(path, 99999.9);
|
||||||
if (time < currentTime) {
|
if (time < currentTime) {
|
||||||
config.set(path, time);
|
config.set(path, time);
|
||||||
@@ -215,27 +375,90 @@ public class ParkourManager {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Löscht NUR die Bestzeiten-Liste (besttimes + names).
|
||||||
|
* Die Strecken-Konfiguration (tracks) bleibt vollständig erhalten.
|
||||||
|
* Wird durch "/nexus parkour clear" aufgerufen.
|
||||||
|
*/
|
||||||
|
public void clearStats() {
|
||||||
|
config.set("besttimes", null);
|
||||||
|
config.set("names", null);
|
||||||
|
save();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void clearStats(int track) {
|
||||||
|
config.set("besttimes." + track, null);
|
||||||
|
save();
|
||||||
|
}
|
||||||
|
|
||||||
public String getTopTen() {
|
public String getTopTen() {
|
||||||
if (!config.contains("besttimes") || config.getConfigurationSection("besttimes") == null)
|
// Fallback: kombiniert beide Tracks (Legacy-Support für %nexuslobby_parkour_top%)
|
||||||
return "§6§l🏆 TOP 10 PARKOUR 🏆\n§7Noch keine Rekorde.";
|
return getTopTen(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getTopTen(int track) {
|
||||||
|
String trackLabel = track == 1 ? " §8(Strecke 1)" : track == 2 ? " §8(Strecke 2)" : "";
|
||||||
|
|
||||||
Map<String, Double> allTimes = new HashMap<>();
|
Map<String, Double> allTimes = new HashMap<>();
|
||||||
for (String uuidStr : config.getConfigurationSection("besttimes").getKeys(false)) {
|
|
||||||
allTimes.put(uuidStr, config.getDouble("besttimes." + uuidStr));
|
if (track == 0) {
|
||||||
|
// Alle Tracks kombiniert (nimmt die beste Zeit je Spieler über alle Tracks)
|
||||||
|
for (int t = 1; t <= 2; t++) {
|
||||||
|
String section = "besttimes." + t;
|
||||||
|
if (config.contains(section) && config.getConfigurationSection(section) != null) {
|
||||||
|
for (String uuidStr : config.getConfigurationSection(section).getKeys(false)) {
|
||||||
|
double time = config.getDouble(section + "." + uuidStr);
|
||||||
|
allTimes.merge(uuidStr, time, Math::min);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
String section = "besttimes." + track;
|
||||||
|
if (!config.contains(section) || config.getConfigurationSection(section) == null) {
|
||||||
|
return "§6§l🏆 TOP 10 PARKOUR" + trackLabel + "\n§7Noch keine Rekorde.";
|
||||||
|
}
|
||||||
|
for (String uuidStr : config.getConfigurationSection(section).getKeys(false)) {
|
||||||
|
allTimes.put(uuidStr, config.getDouble(section + "." + uuidStr));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (allTimes.isEmpty())
|
||||||
|
return "§6§l🏆 TOP 10 PARKOUR" + trackLabel + "\n§7Noch keine Rekorde.";
|
||||||
|
|
||||||
List<Map.Entry<String, Double>> sortedList = allTimes.entrySet().stream()
|
List<Map.Entry<String, Double>> sortedList = allTimes.entrySet().stream()
|
||||||
.sorted(Map.Entry.comparingByValue())
|
.sorted(Map.Entry.comparingByValue())
|
||||||
.limit(10)
|
.limit(10)
|
||||||
.toList();
|
.toList();
|
||||||
|
|
||||||
StringBuilder builder = new StringBuilder("§6§l🏆 TOP 10 PARKOUR 🏆");
|
StringBuilder builder = new StringBuilder("§6§l🏆 TOP 10 PARKOUR" + trackLabel);
|
||||||
int rank = 1;
|
int rank = 1;
|
||||||
for (Map.Entry<String, Double> entry : sortedList) {
|
for (Map.Entry<String, Double> entry : sortedList) {
|
||||||
String name = config.getString("names." + entry.getKey(), "Unbekannt");
|
String name = config.getString("names." + entry.getKey(), "Unbekannt");
|
||||||
builder.append("\n§e#").append(rank).append(" §f").append(name).append(" §8» §a").append(String.format("%.2f", entry.getValue())).append("s");
|
builder.append("\n§e#").append(rank).append(" §f").append(name)
|
||||||
|
.append(" §8» §a").append(String.format("%.2f", entry.getValue())).append("s");
|
||||||
rank++;
|
rank++;
|
||||||
}
|
}
|
||||||
return builder.toString();
|
return builder.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
// Hilfsmethoden
|
||||||
|
// ─────────────────────────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
public boolean isIngame(Player player) { return startTime.containsKey(player.getUniqueId()); }
|
||||||
|
public Location getCheckpoint(Player player) { return lastCheckpointLoc.get(player.getUniqueId()); }
|
||||||
|
|
||||||
|
/** Gibt die Strecken-Info als lesbaren String aus (für Admin-Feedback). */
|
||||||
|
public String getTrackInfo() {
|
||||||
|
StringBuilder sb = new StringBuilder("§8[§6Parkour§8] §7Track-Status:\n");
|
||||||
|
for (int t = 1; t <= 2; t++) {
|
||||||
|
int cps = getOrderedCheckpoints(t).size();
|
||||||
|
boolean ready = isTrackReady(t);
|
||||||
|
sb.append(" §eStrecke ").append(t).append(": ")
|
||||||
|
.append(ready ? "§a✔" : "§c✘")
|
||||||
|
.append(" §7(").append(cps).append(" Checkpoints)");
|
||||||
|
if (t < 2) sb.append("\n");
|
||||||
|
}
|
||||||
|
return sb.toString();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
@@ -146,20 +146,50 @@ public class ServerSwitcherListener implements Listener, CommandExecutor {
|
|||||||
int clickedSlot = e.getSlot();
|
int clickedSlot = e.getSlot();
|
||||||
ConfigurationSection serversSection = NexusLobby.getInstance().getConfig().getConfigurationSection("compass.servers");
|
ConfigurationSection serversSection = NexusLobby.getInstance().getConfig().getConfigurationSection("compass.servers");
|
||||||
|
|
||||||
if (serversSection != null) {
|
if (serversSection == null) return;
|
||||||
for (String serverId : serversSection.getKeys(false)) {
|
|
||||||
String path = "compass.servers." + serverId;
|
|
||||||
int configSlot = NexusLobby.getInstance().getConfig().getInt(path + ".slot", -1);
|
|
||||||
|
|
||||||
if (configSlot == clickedSlot) {
|
for (String serverId : serversSection.getKeys(false)) {
|
||||||
String command = NexusLobby.getInstance().getConfig().getString(path + ".command");
|
String path = "compass.servers." + serverId;
|
||||||
if (command != null && !command.isEmpty()) {
|
int configSlot = NexusLobby.getInstance().getConfig().getInt(path + ".slot", -1);
|
||||||
p.closeInventory();
|
|
||||||
p.chat("/" + command);
|
if (configSlot != clickedSlot) continue;
|
||||||
return;
|
|
||||||
}
|
String command = NexusLobby.getInstance().getConfig().getString(path + ".command", "").trim();
|
||||||
}
|
if (command.isEmpty()) return;
|
||||||
|
|
||||||
|
p.closeInventory();
|
||||||
|
|
||||||
|
// "server <name>" → BungeeCord Connect Plugin-Message
|
||||||
|
if (command.toLowerCase().startsWith("server ")) {
|
||||||
|
String serverName = command.substring(7).trim();
|
||||||
|
connectBungee(p, serverName);
|
||||||
|
} else {
|
||||||
|
// Alle anderen Befehle als Konsolen-Befehl ausführen
|
||||||
|
Bukkit.dispatchCommand(Bukkit.getConsoleSender(),
|
||||||
|
command.replace("%player%", p.getName()));
|
||||||
}
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sendet den Spieler per BungeeCord Plugin-Message an einen anderen Server.
|
||||||
|
*/
|
||||||
|
private void connectBungee(Player player, String serverName) {
|
||||||
|
if (!Bukkit.getMessenger().isOutgoingChannelRegistered(
|
||||||
|
NexusLobby.getInstance(), "BungeeCord")) {
|
||||||
|
player.sendMessage("§8[§6Nexus§8] §cKein BungeeCord-Kanal registriert.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
java.io.ByteArrayOutputStream b = new java.io.ByteArrayOutputStream();
|
||||||
|
java.io.DataOutputStream out = new java.io.DataOutputStream(b);
|
||||||
|
out.writeUTF("Connect");
|
||||||
|
out.writeUTF(serverName);
|
||||||
|
player.sendPluginMessage(NexusLobby.getInstance(), "BungeeCord", b.toByteArray());
|
||||||
|
player.sendMessage("§8[§6Nexus§8] §7Verbinde mit §e" + serverName + "§7...");
|
||||||
|
} catch (java.io.IOException ex) {
|
||||||
|
player.sendMessage("§8[§6Nexus§8] §cFehler beim Verbinden: " + ex.getMessage());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -41,7 +41,7 @@ public class DoubleJump implements Listener {
|
|||||||
@EventHandler
|
@EventHandler
|
||||||
public void onMove(PlayerMoveEvent event) {
|
public void onMove(PlayerMoveEvent event) {
|
||||||
Player player = event.getPlayer();
|
Player player = event.getPlayer();
|
||||||
if (player.getGameMode() != GameMode.CREATIVE && player.getLocation().subtract(0, 0.1, 0).getBlock().getType() != Material.AIR) {
|
if (player.getGameMode() != GameMode.CREATIVE && player.getGameMode() != GameMode.SPECTATOR && player.getLocation().subtract(0, 0.1, 0).getBlock().getType() != Material.AIR) {
|
||||||
if (NexusLobby.getInstance().getConfig().getBoolean("doublejump.enabled", true)) {
|
if (NexusLobby.getInstance().getConfig().getBoolean("doublejump.enabled", true)) {
|
||||||
player.setAllowFlight(true);
|
player.setAllowFlight(true);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ import org.bukkit.event.Listener;
|
|||||||
import org.bukkit.event.block.Action;
|
import org.bukkit.event.block.Action;
|
||||||
import org.bukkit.event.player.PlayerInteractEvent;
|
import org.bukkit.event.player.PlayerInteractEvent;
|
||||||
import org.bukkit.event.player.PlayerJoinEvent;
|
import org.bukkit.event.player.PlayerJoinEvent;
|
||||||
|
import org.bukkit.event.player.PlayerQuitEvent;
|
||||||
import org.bukkit.inventory.ItemStack;
|
import org.bukkit.inventory.ItemStack;
|
||||||
import org.bukkit.inventory.meta.ItemMeta;
|
import org.bukkit.inventory.meta.ItemMeta;
|
||||||
|
|
||||||
@@ -84,4 +85,9 @@ public class PlayerHider implements Listener {
|
|||||||
private String colorize(String s) {
|
private String colorize(String s) {
|
||||||
return s == null ? "" : s.replace("&", "§");
|
return s == null ? "" : s.replace("&", "§");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@EventHandler
|
||||||
|
public void onQuit(PlayerQuitEvent event) {
|
||||||
|
hidden.remove(event.getPlayer().getUniqueId());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
@@ -1,24 +1,18 @@
|
|||||||
package de.nexuslobby.utils;
|
package de.nexuslobby.utils;
|
||||||
|
|
||||||
import com.google.gson.JsonObject;
|
|
||||||
import com.google.gson.JsonParser;
|
|
||||||
import com.google.gson.JsonSyntaxException;
|
|
||||||
import de.nexuslobby.NexusLobby;
|
import de.nexuslobby.NexusLobby;
|
||||||
import org.bukkit.Bukkit;
|
import org.bukkit.Bukkit;
|
||||||
|
|
||||||
import java.io.BufferedReader;
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStreamReader;
|
import java.io.InputStream;
|
||||||
import java.net.HttpURLConnection;
|
|
||||||
import java.net.URL;
|
import java.net.URL;
|
||||||
import java.nio.charset.StandardCharsets;
|
import java.util.Scanner;
|
||||||
import java.util.function.Consumer;
|
import java.util.function.Consumer;
|
||||||
|
|
||||||
public class UpdateChecker {
|
public class UpdateChecker {
|
||||||
|
|
||||||
private final NexusLobby plugin;
|
private final NexusLobby plugin;
|
||||||
private static final String API_URL = "https://git.viper.ipv64.net/api/v1/repos/M_Viper/NexusLobby/releases/latest";
|
private static final int RESOURCE_ID = 132388;
|
||||||
private static final int TIMEOUT_MS = 5000;
|
|
||||||
|
|
||||||
public UpdateChecker(NexusLobby plugin) {
|
public UpdateChecker(NexusLobby plugin) {
|
||||||
this.plugin = plugin;
|
this.plugin = plugin;
|
||||||
@@ -26,60 +20,17 @@ public class UpdateChecker {
|
|||||||
|
|
||||||
public void getVersion(final Consumer<String> consumer) {
|
public void getVersion(final Consumer<String> consumer) {
|
||||||
Bukkit.getScheduler().runTaskAsynchronously(this.plugin, () -> {
|
Bukkit.getScheduler().runTaskAsynchronously(this.plugin, () -> {
|
||||||
HttpURLConnection connection = null;
|
try (InputStream is = new URL(
|
||||||
try {
|
"https://api.spigotmc.org/legacy/update.php?resource=" + RESOURCE_ID + "/~").openStream();
|
||||||
URL url = new URL(API_URL);
|
Scanner scanner = new Scanner(is)) {
|
||||||
connection = (HttpURLConnection) url.openConnection();
|
|
||||||
connection.setRequestMethod("GET");
|
|
||||||
connection.setConnectTimeout(TIMEOUT_MS);
|
|
||||||
connection.setReadTimeout(TIMEOUT_MS);
|
|
||||||
connection.setRequestProperty("Accept", "application/json");
|
|
||||||
connection.setRequestProperty("User-Agent", "NexusLobby-UpdateChecker");
|
|
||||||
|
|
||||||
int responseCode = connection.getResponseCode();
|
if (scanner.hasNext()) {
|
||||||
if (responseCode != HttpURLConnection.HTTP_OK) {
|
consumer.accept(scanner.next());
|
||||||
plugin.getLogger().warning("Update-Check fehlgeschlagen: HTTP " + responseCode);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try (BufferedReader reader = new BufferedReader(
|
|
||||||
new InputStreamReader(connection.getInputStream(), StandardCharsets.UTF_8))) {
|
|
||||||
|
|
||||||
StringBuilder response = new StringBuilder();
|
|
||||||
String line;
|
|
||||||
while ((line = reader.readLine()) != null) {
|
|
||||||
response.append(line);
|
|
||||||
}
|
|
||||||
|
|
||||||
parseAndNotify(response.toString(), consumer);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
plugin.getLogger().warning("Update-Check fehlgeschlagen: " + e.getMessage());
|
plugin.getLogger().warning("Update-Check fehlgeschlagen: " + e.getMessage());
|
||||||
} finally {
|
|
||||||
if (connection != null) {
|
|
||||||
connection.disconnect();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private void parseAndNotify(String jsonResponse, Consumer<String> consumer) {
|
|
||||||
try {
|
|
||||||
JsonObject json = JsonParser.parseString(jsonResponse).getAsJsonObject();
|
|
||||||
|
|
||||||
if (json.has("tag_name")) {
|
|
||||||
String version = json.get("tag_name").getAsString();
|
|
||||||
// Entferne 'v' Präfix falls vorhanden (z.B. v1.0.0 -> 1.0.0)
|
|
||||||
if (version.startsWith("v")) {
|
|
||||||
version = version.substring(1);
|
|
||||||
}
|
|
||||||
consumer.accept(version);
|
|
||||||
} else {
|
|
||||||
plugin.getLogger().warning("Update-Check: 'tag_name' nicht in API-Response gefunden");
|
|
||||||
}
|
|
||||||
} catch (JsonSyntaxException e) {
|
|
||||||
plugin.getLogger().warning("Update-Check: Ungültiges JSON-Format - " + e.getMessage());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
@@ -3,6 +3,8 @@ package de.nexuslobby.utils;
|
|||||||
import de.nexuslobby.NexusLobby;
|
import de.nexuslobby.NexusLobby;
|
||||||
import org.bukkit.Bukkit;
|
import org.bukkit.Bukkit;
|
||||||
import org.bukkit.Location;
|
import org.bukkit.Location;
|
||||||
|
import org.bukkit.World;
|
||||||
|
import org.bukkit.configuration.file.FileConfiguration;
|
||||||
import org.bukkit.entity.Player;
|
import org.bukkit.entity.Player;
|
||||||
import org.bukkit.event.EventHandler;
|
import org.bukkit.event.EventHandler;
|
||||||
import org.bukkit.event.Listener;
|
import org.bukkit.event.Listener;
|
||||||
@@ -17,7 +19,21 @@ public class VoidProtection implements Listener {
|
|||||||
Player player = event.getPlayer();
|
Player player = event.getPlayer();
|
||||||
if (player.getLocation().getY() < 0) {
|
if (player.getLocation().getY() < 0) {
|
||||||
if (NexusLobby.getInstance().getConfig().getBoolean("void_protection.teleport_to_spawn", true)) {
|
if (NexusLobby.getInstance().getConfig().getBoolean("void_protection.teleport_to_spawn", true)) {
|
||||||
Location spawn = player.getWorld().getSpawnLocation();
|
FileConfiguration config = NexusLobby.getInstance().getConfig();
|
||||||
|
Location spawn;
|
||||||
|
if (config.contains("spawn.world")) {
|
||||||
|
String worldName = config.getString("spawn.world");
|
||||||
|
World world = Bukkit.getWorld(worldName);
|
||||||
|
if (world != null) {
|
||||||
|
spawn = new Location(world,
|
||||||
|
config.getDouble("spawn.x"), config.getDouble("spawn.y"), config.getDouble("spawn.z"),
|
||||||
|
(float) config.getDouble("spawn.yaw"), (float) config.getDouble("spawn.pitch"));
|
||||||
|
} else {
|
||||||
|
spawn = player.getWorld().getSpawnLocation();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
spawn = player.getWorld().getSpawnLocation();
|
||||||
|
}
|
||||||
player.teleport(spawn);
|
player.teleport(spawn);
|
||||||
String msg = de.nexuslobby.utils.LangManager.get("void_protection_message");
|
String msg = de.nexuslobby.utils.LangManager.get("void_protection_message");
|
||||||
if (msg != null && !msg.isEmpty()) {
|
if (msg != null && !msg.isEmpty()) {
|
||||||
|
|||||||
@@ -482,6 +482,5 @@ ball:
|
|||||||
#
|
#
|
||||||
# Benötigst du Hilfe oder Support?
|
# Benötigst du Hilfe oder Support?
|
||||||
# - Webseite: https://m-viper.de
|
# - Webseite: https://m-viper.de
|
||||||
# - Dokumentation: https://git.viper.ipv64.net/M_Viper/NexusLobby/wiki
|
# - Dokumentation: https://viper-network.de/wiki/NexusLobby/
|
||||||
# - Discord: https://discord.com/invite/FdRs4BRd8D
|
# - Discord: https://discord.com/invite/FdRs4BRd8D
|
||||||
# - GitHub: https://git.viper.ipv64.net/M_Viper/NexusLobby
|
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
name: NexusLobby
|
name: NexusLobby
|
||||||
main: de.nexuslobby.NexusLobby
|
main: de.nexuslobby.NexusLobby
|
||||||
version: "1.1.3"
|
version: "1.9"
|
||||||
api-version: "1.21"
|
api-version: "1.21"
|
||||||
author: M_Viper
|
author: M_Viper
|
||||||
description: Modular Lobby Plugin with an invisible Particle-Parkour system.
|
description: Modular Lobby Plugin with an invisible Particle-Parkour system.
|
||||||
|
|||||||
Reference in New Issue
Block a user