3 Commits
1.1 ... 1.3

Author SHA1 Message Date
710f8cf4f5 README.md aktualisiert 2026-01-30 16:25:29 +00:00
67b06252cb Update from Git Manager GUI 2026-01-30 17:17:16 +01:00
4e8705ba8e Upload pom.xml via GUI 2026-01-30 16:17:15 +00:00
6 changed files with 240 additions and 104 deletions

View File

@@ -5,6 +5,7 @@
![Version](https://img.shields.io/badge/Version-1.18.x--1.21.x-green.svg) ![Version](https://img.shields.io/badge/Version-1.18.x--1.21.x-green.svg)
![Java](https://img.shields.io/badge/Java-17+-orange.svg) ![Java](https://img.shields.io/badge/Java-17+-orange.svg)
![Type](https://img.shields.io/badge/Type-Utility-blue.svg) ![Type](https://img.shields.io/badge/Type-Utility-blue.svg)
![Updates](https://img.shields.io/badge/Updates-Git--Notifier-gold.svg)
--- ---
@@ -14,25 +15,25 @@
* **📱 Smart-GUI Interface:** Ein Rechtsklick auf das Modul öffnet ein elegantes Menü zur Direktauswahl aller Etagen im Schacht. * **📱 Smart-GUI Interface:** Ein Rechtsklick auf das Modul öffnet ein elegantes Menü zur Direktauswahl aller Etagen im Schacht.
* **💎 Dynamische Hologramme:** Nutzt moderne **Text-Displays** zur Anzeige der Etagen-Namen direkt über dem Modul (ressourcenschonend). * **💎 Dynamische Hologramme:** Nutzt moderne **Text-Displays** zur Anzeige der Etagen-Namen direkt über dem Modul (ressourcenschonend).
* **🔒 Privatsphäre-System:** Schalte Etagen per Befehl zwischen `Public` (öffentlich) und `Private` (nur Besitzer) um. * **🔒 Privatsphäre-System:** Schalte Etagen per Befehl zwischen `Public` (öffentlich) und `Private` (nur Besitzer) um.
* **🛠 Voll anpassbares Rezept:** Erstelle dein eigenes Crafting-Layout inklusive frei wählbarer Materialien direkt in der Config.
* **🎨 Customizing:** Volle Unterstützung von **Farbcodes & Hex-Farben** für individuelle Etagennamen. * **🎨 Customizing:** Volle Unterstützung von **Farbcodes & Hex-Farben** für individuelle Etagennamen.
* **🛡️ Grief-Schutz:** Nur Besitzer oder Admins können Module konfigurieren oder entfernen. * **🛡️ Grief-Schutz:** Nur Besitzer oder Admins können Module konfigurieren oder entfernen.
* **📝 Auto-Schilder:** Erstelle stylische Anzeigen automatisch mit `[Elevator]` in der ersten Zeile eines Schildes. * **🔔 Ingame Update-Check:** Admins erhalten beim Joinen eine interaktive Meldung mit direktem Download-Link, sobald eine neue Version auf Git verfügbar ist.
--- ---
## 🛠 Crafting-Rezept ## 🛠 Crafting-System
Das **Premium Aufzug-Modul** kann an jeder Werkbank mit folgendem Rezept hergestellt werden: Dank des neuen dynamischen Rezept-Systems kannst du das **Aufzug-Modul** in der `config.yml` völlig frei gestalten. Standardmäßig ist folgendes Layout vorkonfiguriert:
| Slot | Material | | Slot | Position | Standard-Material |
| :--- | :--- | |:-----|:---------|:------------------|
| **Oben Links** | Tageslichtsensor | | **1, 3, 7, 9** | Ecken | Eisenbarren |
| **Oben Rechts** | Tageslichtsensor | | **2, 4, 6, 8** | Seiten | Enderperle |
| **Mitte** | Eisenblock | | **5** | Mitte | Tageslichtsensor |
| **Unten Links** | Tageslichtsensor |
| **Unten Rechts** | Tageslichtsensor |
**Ergebnis:** `1x Aufzug-Modul` > **Hinweis:** Das Gitter (Shape) und die Zutaten (Ingredients) können jederzeit ohne Code-Änderungen in der Konfiguration angepasst werden.
> **Material-Liste:** Alle verfügbaren Materialien findest du in der [Spigot Material Documentation](https://hub.spigotmc.org/javadocs/spigot/org/bukkit/Material.html).
--- ---
@@ -41,7 +42,7 @@ Das **Premium Aufzug-Modul** kann an jeder Werkbank mit folgendem Rezept hergest
### Befehle ### Befehle
| Befehl | Beschreibung | Berechtigung | | Befehl | Beschreibung | Berechtigung |
| :--- | :--- | :--- | |:-------|:-------------|:-------------|
| `/elevator name <Text>` | Ändert den Namen der aktuellen Etage. | `elevator.use` | | `/elevator name <Text>` | Ändert den Namen der aktuellen Etage. | `elevator.use` |
| `/elevator private` | Setzt die Etage auf **Privat**. | `elevator.use` | | `/elevator private` | Setzt die Etage auf **Privat**. | `elevator.use` |
| `/elevator public` | Setzt die Etage auf **Öffentlich**. | `elevator.use` | | `/elevator public` | Setzt die Etage auf **Öffentlich**. | `elevator.use` |
@@ -49,40 +50,48 @@ Das **Premium Aufzug-Modul** kann an jeder Werkbank mit folgendem Rezept hergest
### Administration ### Administration
| Berechtigung | Beschreibung | | Berechtigung | Beschreibung |
| :--- | :--- | |:-------------|:-------------|
| `elevator.admin` | Verwalten & Abbauen fremder Aufzüge sowie Bypass von Private-Sperren. | | `elevator.admin` | Verwalten & Abbauen fremder Aufzüge sowie Erhalt von Update-Benachrichtigungen. |
--- ---
## ⚙️ Konfiguration (Auszug) ## ⚙️ Konfiguration (Auszug)
```yaml ```yaml
settings: # Dynamisches Rezept-Beispiel
max-distance: 64 # Maximale Distanz zwischen Modulen recipe:
cooldown: 500 # Cooldown in ms zwischen Teleports shape:
- "ABA"
- "BCB"
- "ABA"
ingredients:
A: "IRON_INGOT"
B: "ENDER_PEARL"
C: "DAYLIGHT_DETECTOR"
# Visuelle Einstellungen
visuals: visuals:
enable-particles: true enable-particles: true
hologram-text: "&8&l» &b&lEtage %floor% &8&l«" hologram-text: "&8&l» &b&lEtage %floor% &8&l«"
hologram-scale: 1.5 hologram-scale: 1.5
``` ```
---
## 🏗 Installation ## 🏗 Installation
1. **Download:** Lade die `Elevator.jar` herunter. 1. **Download**: Lade die neueste `Elevator.jar` von unserem Repository herunter.
2. **Upload:** Verschiebe die Datei in deinen `/plugins` Ordner. 2. **Upload**: Verschiebe die Datei in deinen `/plugins` Ordner.
3. **Start:** Starte deinen Server neu, damit das Plugin geladen wird. 3. **Start**: Starte deinen Server neu, damit das Plugin geladen wird.
4. **Setup:** Passe die `config.yml` im Ordner `/plugins/Elevator/` nach deinen Wünschen an und lade das Plugin ggf. neu. 4. **Setup**: Passe die `config.yml` im Ordner `/plugins/Elevator/` nach deinen Wünschen an.
--- ---
## 👤 Autor & Support ## 👤 Autor & Support
**Erstellt von: M_Viper** **Erstellt von:** M_Viper
💡 Entwickelt mit Fokus auf **Performance**, **Design** und **maximalen Bedienkomfort**. Bei Fragen, Fehlern oder Verbesserungsvorschlägen tritt gerne unserem Discord bei: 💡 Entwickelt mit Fokus auf Performance, Design und maximalen Bedienkomfort. Bei Fragen, Fehlern oder Verbesserungsvorschlägen tritt gerne unserem Discord bei.
[![Discord Support](https://img.shields.io/badge/Discord-Support-7289DA?style=for-the-badge&logo=discord)](https://discord.com/invite/FdRs4BRd8D)
--- ---
*Viper Plugins © 2026 — Effiziente Systemlösungen für deinen Server.*
**Viper Plugins** © 2026 — Effiziente Systemlösungen für deinen Server.

View File

@@ -3,7 +3,7 @@
<modelVersion>4.0.0</modelVersion> <modelVersion>4.0.0</modelVersion>
<groupId>de.mviper</groupId> <groupId>de.mviper</groupId>
<artifactId>Elevator</artifactId> <artifactId>Elevator</artifactId>
<version>1.1</version> <version>1.2</version>
<properties> <properties>
<maven.compiler.source>17</maven.compiler.source> <maven.compiler.source>17</maven.compiler.source>

View File

@@ -1,39 +1,91 @@
package de.mviper.elevator; package de.mviper.elevator;
import net.md_5.bungee.api.chat.ClickEvent;
import net.md_5.bungee.api.chat.HoverEvent;
import net.md_5.bungee.api.chat.TextComponent;
import net.md_5.bungee.api.chat.hover.content.Text;
import org.bukkit.Material; import org.bukkit.Material;
import org.bukkit.NamespacedKey; import org.bukkit.NamespacedKey;
import org.bukkit.configuration.ConfigurationSection;
import org.bukkit.enchantments.Enchantment; import org.bukkit.enchantments.Enchantment;
import org.bukkit.entity.Player;
import org.bukkit.event.EventHandler;
import org.bukkit.event.Listener;
import org.bukkit.event.player.PlayerJoinEvent;
import org.bukkit.inventory.ItemFlag; import org.bukkit.inventory.ItemFlag;
import org.bukkit.inventory.ItemStack; import org.bukkit.inventory.ItemStack;
import org.bukkit.inventory.ShapedRecipe; import org.bukkit.inventory.ShapedRecipe;
import org.bukkit.inventory.meta.ItemMeta; import org.bukkit.inventory.meta.ItemMeta;
import org.bukkit.plugin.java.JavaPlugin; import org.bukkit.plugin.java.JavaPlugin;
import java.util.Arrays; import java.util.Arrays;
import java.util.List;
import java.util.logging.Level;
public class Elevator extends JavaPlugin { public class Elevator extends JavaPlugin implements Listener {
private static Elevator instance; private static Elevator instance;
private DatabaseManager databaseManager; private DatabaseManager databaseManager;
private HologramManager hologramManager; private HologramManager hologramManager;
private String latestVersionFound;
private final String releaseUrl = "https://git.viper.ipv64.net/M_Viper/Elevator/releases";
@Override @Override
public void onEnable() { public void onEnable() {
instance = this; instance = this;
saveDefaultConfig(); saveDefaultConfig();
this.databaseManager = new DatabaseManager(); this.databaseManager = new DatabaseManager();
this.hologramManager = new HologramManager(); this.hologramManager = new HologramManager();
hologramManager.purgeAllHolograms(); hologramManager.purgeAllHolograms();
// Events & Commands registrieren
getServer().getPluginManager().registerEvents(new ElevatorListener(), this); getServer().getPluginManager().registerEvents(new ElevatorListener(), this);
getServer().getPluginManager().registerEvents(this, this);
getCommand("elevator").setExecutor(new ElevatorCommand()); getCommand("elevator").setExecutor(new ElevatorCommand());
// Rezept laden
registerElevatorRecipe(); registerElevatorRecipe();
// Update Check starten
checkForUpdates();
}
private void checkForUpdates() {
new UpdateChecker(this).getLatestVersion(version -> {
this.latestVersionFound = version;
String currentVersion = this.getDescription().getVersion();
if (!currentVersion.equalsIgnoreCase(version)) {
getLogger().info("====================================================");
getLogger().info("NEUES UPDATE VERFÜGBAR: " + version);
getLogger().info("Download: " + releaseUrl);
getLogger().info("====================================================");
}
});
}
@EventHandler
public void onJoin(PlayerJoinEvent event) {
Player player = event.getPlayer();
if (player.hasPermission("elevator.admin") || player.isOp()) {
if (latestVersionFound != null && !this.getDescription().getVersion().equalsIgnoreCase(latestVersionFound)) {
player.sendMessage(" ");
player.sendMessage("§b§l[Elevator] §eEine neue Version (§b" + latestVersionFound + "§e) ist verfügbar!");
TextComponent message = new TextComponent("§eDownload: ");
TextComponent link = new TextComponent("§6§l[KLICK HIER FÜR UPDATE]");
link.setClickEvent(new ClickEvent(ClickEvent.Action.OPEN_URL, releaseUrl));
link.setHoverEvent(new HoverEvent(HoverEvent.Action.SHOW_TEXT, new Text("§7Öffnet die Release-Seite")));
message.addExtra(link);
player.spigot().sendMessage(message);
player.sendMessage(" ");
}
}
} }
/**
* Registriert das Elevator-Modul Rezept basierend auf der Config-Einstellung.
* Unterstützte Rezept-Typen: "standard", "expensive"
*/
private void registerElevatorRecipe() { private void registerElevatorRecipe() {
// Erstelle das Elevator-Modul Item // Das Resultat-Item (Aufzug-Modul)
ItemStack item = new ItemStack(Material.DAYLIGHT_DETECTOR); ItemStack item = new ItemStack(Material.DAYLIGHT_DETECTOR);
ItemMeta meta = item.getItemMeta(); ItemMeta meta = item.getItemMeta();
if (meta != null) { if (meta != null) {
@@ -48,56 +100,71 @@ public class Elevator extends JavaPlugin {
item.setItemMeta(meta); item.setItemMeta(meta);
} }
// Lese Rezept-Typ aus der Config (Standard: "standard") NamespacedKey key = new NamespacedKey(this, "elevator_module");
String recipeType = getConfig().getString("recipe.type", "standard").toLowerCase(); ShapedRecipe recipe = new ShapedRecipe(key, item);
// Erstelle das ShapedRecipe // Shape laden und bereinigen (verhindert den "Length 5" Fehler)
ShapedRecipe recipe = new ShapedRecipe(new NamespacedKey(this, "elevator_module"), item); List<String> shapeLines = getConfig().getStringList("recipe.shape");
if (shapeLines.size() != 3) {
// Wähle das Rezept basierend auf dem Config-Wert getLogger().severe("Rezept konnte nicht geladen werden: 'shape' benötigt genau 3 Zeilen!");
switch (recipeType) { return;
case "expensive": }
// Teures Rezept:
// D E D D = DIAMOND String[] cleanedRows = new String[3];
// E I E E = ENDER_PEARL for (int i = 0; i < 3; i++) {
// D E D I = IRON_BLOCK // Entferne alle Leerzeichen für die interne Verarbeitung
recipe.shape("DED", "EIE", "DED"); String line = shapeLines.get(i).replace(" ", "");
recipe.setIngredient('D', Material.DIAMOND);
recipe.setIngredient('E', Material.ENDER_PEARL); // Mit Leerzeichen auffüllen, falls die Zeile kürzer als 3 ist
recipe.setIngredient('I', Material.IRON_BLOCK); while (line.length() < 3) {
getLogger().info("§aElevator-Rezept aktiviert: §eExpensive (Diamanten + Enderperlen)"); line += " ";
break; }
// Kürzen, falls sie länger als 3 ist
case "standard": if (line.length() > 3) {
default: line = line.substring(0, 3);
// Standard-Rezept: }
// S . S S = DAYLIGHT_DETECTOR cleanedRows[i] = line;
// . I . I = IRON_BLOCK }
// S . S
recipe.shape("S.S", ".I.", "S.S"); recipe.shape(cleanedRows[0], cleanedRows[1], cleanedRows[2]);
recipe.setIngredient('S', Material.DAYLIGHT_DETECTOR);
recipe.setIngredient('I', Material.IRON_BLOCK); // Zutaten zuweisen
getLogger().info("§aElevator-Rezept aktiviert: §eStandard (Daylight Detector + Iron Block)"); ConfigurationSection section = getConfig().getConfigurationSection("recipe.ingredients");
break; if (section == null) {
getLogger().severe("Keine 'recipe.ingredients' in der Config gefunden!");
return;
}
for (String row : cleanedRows) {
for (char c : row.toCharArray()) {
if (c == ' ') continue;
String matName = section.getString(String.valueOf(c));
if (matName != null) {
Material mat = Material.matchMaterial(matName.toUpperCase());
if (mat != null) {
recipe.setIngredient(c, mat);
} else {
getLogger().warning("Ungültiges Material '" + matName + "' für Symbol '" + c + "'!");
return;
}
} else {
getLogger().warning("Symbol '" + c + "' im Shape hat keine Zuweisung in ingredients!");
return;
}
}
} }
// Rezept zum Server hinzufügen
try { try {
getServer().removeRecipe(key);
getServer().addRecipe(recipe); getServer().addRecipe(recipe);
getLogger().info("Elevator: Crafting-Rezept erfolgreich registriert.");
} catch (Exception e) { } catch (Exception e) {
getLogger().warning("§cKonnte Elevator-Rezept nicht registrieren: " + e.getMessage()); getLogger().log(Level.WARNING, "Fehler beim Rezept-Setup: {0}", e.getMessage());
} }
} }
public static Elevator getInstance() { public static Elevator getInstance() { return instance; }
return instance; public DatabaseManager getDatabaseManager() { return databaseManager; }
} public HologramManager getHologramManager() { return hologramManager; }
public DatabaseManager getDatabaseManager() {
return databaseManager;
}
public HologramManager getHologramManager() {
return hologramManager;
}
} }

View File

@@ -0,0 +1,43 @@
package de.mviper.elevator;
import com.google.gson.JsonObject;
import com.google.gson.JsonParser;
import org.bukkit.Bukkit;
import java.io.InputStreamReader;
import java.net.HttpURLConnection;
import java.net.URL;
import java.util.function.Consumer;
public class UpdateChecker {
private final Elevator plugin;
private final String versionUrl = "https://git.viper.ipv64.net/api/v1/repos/M_Viper/Elevator/releases/latest";
public UpdateChecker(Elevator plugin) {
this.plugin = plugin;
}
public void getLatestVersion(Consumer<String> consumer) {
Bukkit.getScheduler().runTaskAsynchronously(this.plugin, () -> {
try {
HttpURLConnection connection = (HttpURLConnection) new URL(versionUrl).openConnection();
connection.setRequestMethod("GET");
connection.setConnectTimeout(5000);
connection.setReadTimeout(5000);
if (connection.getResponseCode() == 200) {
InputStreamReader reader = new InputStreamReader(connection.getInputStream());
JsonObject jsonResponse = JsonParser.parseReader(reader).getAsJsonObject();
// Holt den Tag-Namen (z.B. v1.1) aus der Gitea API
if (jsonResponse.has("tag_name")) {
String latestVersion = jsonResponse.get("tag_name").getAsString();
consumer.accept(latestVersion);
}
}
} catch (Exception e) {
plugin.getLogger().warning("Update-Check fehlgeschlagen: " + e.getMessage());
}
});
}
}

View File

@@ -1,4 +1,8 @@
# Elevator Config by mviper # =============================================================
# Elevator Config by M_Viper
# =============================================================
# Datenbank-Einstellungen (SQLite wird standardmäßig genutzt, falls false)
mysql: mysql:
enable: false enable: false
host: "localhost" host: "localhost"
@@ -7,51 +11,64 @@ mysql:
user: "root" user: "root"
password: "" password: ""
# Allgemeine Aufzug-Logik
settings: settings:
max-distance: 64 max-distance: 64 # Maximale Reichweite zwischen zwei Modulen
hologram-duration: 45 hologram-duration: 45 # Wie lange das Hologramm (in Ticks) sichtbar bleibt
cooldown: 500 cooldown: 500 # Wartezeit zwischen Teleports (in Millisekunden)
# Optische Effekte & Hologramme
visuals: visuals:
enable-particles: true enable-particles: true
particle-type: "FIREWORKS_SPARK" particle-type: "FIREWORKS_SPARK"
# Nutze HIER nur das & Zeichen für Farben!
# Texte & Farben (Nutze '&' für Farbcodes)
hologram-text: "&8&l» &b&lEtage %floor% &8&l«" hologram-text: "&8&l» &b&lEtage %floor% &8&l«"
actionbar-text: "&fTransport: &b&lEtage %floor%"
# Darstellung des Hologramms
hologram-height-offset: 2.2 hologram-height-offset: 2.2
hologram-scale: 1.5 hologram-scale: 1.5
actionbar-text: "&fTransport: &b&lEtage %floor%"
# Hintergrund-Box (RGBA Werte)
hologram-background-color: hologram-background-color:
alpha: 120 alpha: 120
red: 0 red: 0
green: 0 green: 0
blue: 0 blue: 0
# Akustisches Feedback
sounds: sounds:
enable: true enable: true
type: "BLOCK_NOTE_BLOCK_CHIME" type: "BLOCK_NOTE_BLOCK_CHIME"
volume: 1.0 volume: 1.0
pitch: 1.5 pitch: 1.5
# Nachrichten-System
messages: messages:
prefix: "&8[&bElevator&8] " prefix: "&8[&bElevator&8] "
registered: "&aModul erfolgreich registriert!" registered: "&aModul erfolgreich registriert!"
no-target: "&cKeine weitere Etage gefunden." no-target: "&cKeine weitere Etage gefunden."
# ============================================ # =============================================================
# REZEPT-AUSWAHL # REZEPT-EINSTELLUNGEN
# ============================================ # =============================================================
# Hier kannst du zwischen verschiedenen Crafting-Rezepten wählen. # Crafting-Gitter Übersicht:
# Mögliche Werte: "standard", "expensive" # [ 1 | 2 | 3 ] -> Zeile 1
# # [ 4 | 5 | 6 ] -> Zeile 2
Standard-Rezept: # [ 7 | 8 | 9 ] -> Zeile 3
S.S S = DAYLIGHT_DETECTOR # =============================================================
.I. I = IRON_BLOCK
S.S
#
Teures Rezept (Expensive):
DED D = DIAMOND
EIE E = ENDER_PEARL
DED I = IRON_BLOCK
recipe: recipe:
type: "standard" # Das 3x3 Layout (Maximal 3 Zeichen pro Zeile!)
# Nutze einen Punkt (.) für ein leeres Feld.
shape:
- "ABA" # Slot 1 2 3
- "BCB" # Slot 4 5 6
- "ABA" # Slot 7 8 9
# Material-Zuweisung
# Liste aller Materialien: https://hub.spigotmc.org/javadocs/spigot/org/bukkit/Material.html
ingredients:
A: "IRON_INGOT"
B: "ENDER_PEARL"
C: "DAYLIGHT_DETECTOR"

View File

@@ -1,5 +1,5 @@
name: Elevator name: Elevator
version: 1.1 version: 1.2
main: de.mviper.elevator.Elevator main: de.mviper.elevator.Elevator
api-version: 1.20 api-version: 1.20
author: mviper author: mviper