mirror of
https://github.com/KoopaCode/ArmorStand-Storage.git
synced 2025-04-02 22:17:32 +00:00
Public Release
This commit is contained in:
commit
edee44630c
90
.github/workflows/build.yml
vendored
Normal file
90
.github/workflows/build.yml
vendored
Normal file
@ -0,0 +1,90 @@
|
||||
name: Build
|
||||
on:
|
||||
push:
|
||||
branches: [ "main" ]
|
||||
pull_request:
|
||||
branches: [ "main" ]
|
||||
release:
|
||||
types: [created]
|
||||
|
||||
jobs:
|
||||
build:
|
||||
runs-on: ubuntu-latest
|
||||
permissions:
|
||||
contents: read
|
||||
packages: write
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
|
||||
- name: Set up JDK 17
|
||||
uses: actions/setup-java@v4
|
||||
with:
|
||||
java-version: '17'
|
||||
distribution: 'temurin'
|
||||
cache: maven
|
||||
|
||||
- name: Get Project Info
|
||||
run: |
|
||||
echo "PLUGIN_VERSION=$(mvn help:evaluate -Dexpression=project.version -q -DforceStdout)" >> $GITHUB_ENV
|
||||
echo "PLUGIN_NAME=$(mvn help:evaluate -Dexpression=project.name -q -DforceStdout)" >> $GITHUB_ENV
|
||||
echo "PLUGIN_FILE=$(mvn help:evaluate -Dexpression=project.artifactId -q -DforceStdout)" >> $GITHUB_ENV
|
||||
|
||||
- name: Build with Maven
|
||||
run: mvn -B package --file pom.xml
|
||||
|
||||
- name: Debug Directory
|
||||
run: ls -la target/
|
||||
|
||||
- name: Upload Artifact
|
||||
uses: actions/upload-artifact@v4
|
||||
with:
|
||||
name: ${{ env.PLUGIN_FILE }}-${{ env.PLUGIN_VERSION }}
|
||||
path: target/${{ env.PLUGIN_FILE }}-${{ env.PLUGIN_VERSION }}.jar
|
||||
retention-days: 5
|
||||
|
||||
- name: Send Build Notification
|
||||
if: always()
|
||||
uses: sarisia/actions-status-discord@v1
|
||||
with:
|
||||
webhook: ${{ secrets.DISCORD_WEBHOOK }}
|
||||
title: "${{ job.status == 'success' && '✅ Build Success!' || '❌ Build Failed!' }}"
|
||||
description: |
|
||||
**${{ env.PLUGIN_NAME }} v${{ env.PLUGIN_VERSION }}**
|
||||
By ${{ github.actor }} • ${{ github.sha }}
|
||||
${{ github.repository }}
|
||||
color: ${{ job.status == 'success' && '0x00ff00' || '0xff0000' }}
|
||||
username: "🏺Artifact Build's"
|
||||
avatar_url: "https://github.githubassets.com/images/modules/logos_page/GitHub-Mark.png"
|
||||
|
||||
release:
|
||||
needs: build
|
||||
if: github.event_name == 'release'
|
||||
runs-on: ubuntu-latest
|
||||
permissions:
|
||||
contents: write
|
||||
|
||||
steps:
|
||||
- name: Download Artifact
|
||||
uses: actions/download-artifact@v4
|
||||
with:
|
||||
name: ${{ env.PLUGIN_FILE }}-${{ env.PLUGIN_VERSION }}
|
||||
|
||||
- name: Upload Release
|
||||
uses: softprops/action-gh-release@v2
|
||||
with:
|
||||
files: ${{ env.PLUGIN_FILE }}-${{ env.PLUGIN_VERSION }}.jar
|
||||
fail_on_unmatched_files: true
|
||||
|
||||
- name: Send Release Notification
|
||||
if: always()
|
||||
uses: sarisia/actions-status-discord@v1
|
||||
with:
|
||||
webhook: ${{ secrets.DISCORD_WEBHOOK }}
|
||||
title: "${{ job.status == 'success' && '🎉 Release Published!' || '❌ Release Failed!' }}"
|
||||
description: |
|
||||
**${{ env.PLUGIN_NAME }} v${{ env.PLUGIN_VERSION }}**
|
||||
${{ job.status == 'success' && '➜ https://github.com/${{ github.repository }}/releases/latest' || '' }}
|
||||
color: ${{ job.status == 'success' && '0x00ff00' || '0xff0000' }}
|
||||
username: "🏺Artifact Build's"
|
||||
avatar_url: "https://github.githubassets.com/images/modules/logos_page/GitHub-Mark.png"
|
125
README.md
Normal file
125
README.md
Normal file
@ -0,0 +1,125 @@
|
||||
|
||||
|
||||

|
||||
|
||||
🛡️ Armor Stand Storage v1.0
|
||||
============================
|
||||
<p align="center"> <img src="https://lapislabs.dev/images/inv.png" alt="Storage Image"> </p>
|
||||
|
||||
Transform your armor stands into functional storage units while maintaining their display capabilities! Perfect for kits, storage rooms, and displays.
|
||||
|
||||
**Version:** 1.0
|
||||
|
||||
**Minecraft Versions:** 1.8 - 1.21.x
|
||||
|
||||
**Release Date:** 1/27/25
|
||||
|
||||
✨ Features
|
||||
----------
|
||||
|
||||
* **Smart Storage System**
|
||||
* Shift + Right-Click any armor stand to access storage
|
||||
* 27 slots of storage space per armor stand
|
||||
* Items in armor slots automatically display on the stand
|
||||
* Regular right-click still works for vanilla interactions
|
||||
* **Intelligent Armor Detection**
|
||||
* Automatically detects and displays armor pieces
|
||||
* Works with any armor type (leather, iron, gold, diamond, netherite)
|
||||
* Supports modded armor that follows naming conventions
|
||||
* Compatible with Elytra and player heads
|
||||
* **Server Friendly**
|
||||
* Multi-world support with configurable blocked worlds
|
||||
* Minimal performance impact
|
||||
* SQLite database for reliable storage
|
||||
* Works on Minecraft 1.8+
|
||||
|
||||
🎮 Usage
|
||||
--------
|
||||
|
||||
1. Place an armor stand
|
||||
2. Add armor/items normally OR
|
||||
3. Shift + Right-Click to open storage
|
||||
4. Place items anywhere in the inventory:
|
||||
* Armor pieces will automatically display
|
||||
* Other items are safely stored
|
||||
* First 6 slots are prioritized for display
|
||||
|
||||
⚙️ Configuration
|
||||
----------------
|
||||
|
||||
# Disable storage in specific worlds
|
||||
disabled-worlds:
|
||||
#- spawn_world
|
||||
#- event_world
|
||||
|
||||
# Debug mode for detailed logging
|
||||
debug: false
|
||||
|
||||
# Storage settings
|
||||
storage:
|
||||
inventory-title: "Armor Stand Storage"
|
||||
inventory-rows: 3
|
||||
|
||||
🔒 Permissions
|
||||
--------------
|
||||
|
||||
* `armorstandstorage.use` - Access to armor stand storage (default: op)
|
||||
|
||||
📋 Commands
|
||||
-----------
|
||||
|
||||
No commands - just shift-right-click to use!
|
||||
|
||||
💡 Tips
|
||||
-------
|
||||
|
||||
* Place armor in any slot - it will automatically display
|
||||
* Use regular right-click for vanilla armor stand features
|
||||
* Items are saved even if the armor stand breaks
|
||||
* Works great for shop displays and storage rooms
|
||||
* Perfect for RPG servers and town builds
|
||||
|
||||
🔧 Installation
|
||||
---------------
|
||||
|
||||
1. Download the plugin
|
||||
2. Place in your plugins folder
|
||||
3. Restart server
|
||||
4. Configure (optional)
|
||||
5. Start using!
|
||||
|
||||
🎯 Planned Features
|
||||
-------------------
|
||||
|
||||
* Custom inventory sizes
|
||||
* Per-world inventory settings
|
||||
* Hologram support
|
||||
* Shop integration
|
||||
* API for developers
|
||||
|
||||
🆕 What's New in 1.0
|
||||
--------------------
|
||||
|
||||
* Initial release
|
||||
* Full armor stand storage functionality
|
||||
* Multi-world support
|
||||
* Automatic armor detection and display
|
||||
* SQLite database integration
|
||||
* Legacy version support (1.8+)
|
||||
|
||||
📝 Version History
|
||||
------------------
|
||||
|
||||
* **1.0** (1.27.2025)
|
||||
* Initial release
|
||||
* Core functionality implemented
|
||||
* Database storage system
|
||||
* Multi-version support
|
||||
|
||||
ArmorStandStorage v1.0 - © 2024 Koopa
|
||||
|
||||
Made with ❤️ by Koopa
|
||||
|
||||
[GitHub Issues](https://github.com/KoopaCode/ArmorStand-Storage/issues) | [Discord](https://discord.gg/KmHGjaHWct)
|
||||
|
||||
`storage` `free` `armorstand` `plugin` `1.8-1.21`
|
83
pom.xml
Normal file
83
pom.xml
Normal file
@ -0,0 +1,83 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<groupId>com.koopacraft</groupId>
|
||||
<artifactId>armorstandstorage</artifactId>
|
||||
<version>1.0-SNAPSHOT</version>
|
||||
<packaging>jar</packaging>
|
||||
|
||||
<name>ArmorStandStorage</name>
|
||||
<description>A plugin that allows storing items in armor stands</description>
|
||||
|
||||
<properties>
|
||||
<java.version>17</java.version>
|
||||
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
|
||||
</properties>
|
||||
|
||||
<repositories>
|
||||
<repository>
|
||||
<id>papermc</id>
|
||||
<url>https://repo.papermc.io/repository/maven-public/</url>
|
||||
</repository>
|
||||
</repositories>
|
||||
|
||||
<dependencies>
|
||||
<dependency>
|
||||
<groupId>io.papermc.paper</groupId>
|
||||
<artifactId>paper-api</artifactId>
|
||||
<version>1.21.1-R0.1-SNAPSHOT</version>
|
||||
<scope>provided</scope>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>org.xerial</groupId>
|
||||
<artifactId>sqlite-jdbc</artifactId>
|
||||
<version>3.45.1.0</version>
|
||||
<scope>compile</scope>
|
||||
</dependency>
|
||||
</dependencies>
|
||||
|
||||
<build>
|
||||
<plugins>
|
||||
<plugin>
|
||||
<groupId>org.apache.maven.plugins</groupId>
|
||||
<artifactId>maven-compiler-plugin</artifactId>
|
||||
<version>3.11.0</version>
|
||||
<configuration>
|
||||
<source>${java.version}</source>
|
||||
<target>${java.version}</target>
|
||||
</configuration>
|
||||
</plugin>
|
||||
<plugin>
|
||||
<groupId>org.apache.maven.plugins</groupId>
|
||||
<artifactId>maven-shade-plugin</artifactId>
|
||||
<version>3.5.1</version>
|
||||
<executions>
|
||||
<execution>
|
||||
<phase>package</phase>
|
||||
<goals>
|
||||
<goal>shade</goal>
|
||||
</goals>
|
||||
<configuration>
|
||||
<createDependencyReducedPom>false</createDependencyReducedPom>
|
||||
<relocations>
|
||||
<relocation>
|
||||
<pattern>org.sqlite</pattern>
|
||||
<shadedPattern>com.koopacraft.armorstandstorage.lib.sqlite</shadedPattern>
|
||||
</relocation>
|
||||
</relocations>
|
||||
</configuration>
|
||||
</execution>
|
||||
</executions>
|
||||
</plugin>
|
||||
</plugins>
|
||||
<resources>
|
||||
<resource>
|
||||
<directory>src/main/resources</directory>
|
||||
<filtering>true</filtering>
|
||||
</resource>
|
||||
</resources>
|
||||
</build>
|
||||
</project>
|
@ -0,0 +1,342 @@
|
||||
package com.koopacraft.armorstandstorage;
|
||||
|
||||
import org.bukkit.Bukkit;
|
||||
import org.bukkit.entity.ArmorStand;
|
||||
import org.bukkit.entity.Entity;
|
||||
import org.bukkit.entity.EntityType;
|
||||
import org.bukkit.entity.Player;
|
||||
import org.bukkit.event.EventHandler;
|
||||
import org.bukkit.event.Listener;
|
||||
import org.bukkit.event.entity.EntityDeathEvent;
|
||||
import org.bukkit.event.player.PlayerInteractAtEntityEvent;
|
||||
import org.bukkit.event.entity.CreatureSpawnEvent;
|
||||
import org.bukkit.inventory.Inventory;
|
||||
import org.bukkit.inventory.ItemStack;
|
||||
import org.bukkit.event.inventory.InventoryCloseEvent;
|
||||
import org.bukkit.event.inventory.InventoryClickEvent;
|
||||
import org.bukkit.event.inventory.InventoryDragEvent;
|
||||
import org.bukkit.event.inventory.InventoryType;
|
||||
import org.bukkit.event.inventory.InventoryAction;
|
||||
import org.bukkit.inventory.EntityEquipment;
|
||||
import org.bukkit.event.player.PlayerArmorStandManipulateEvent;
|
||||
|
||||
import java.util.HashMap;
|
||||
import java.util.UUID;
|
||||
|
||||
public class ArmorStandListener implements Listener {
|
||||
private final ArmorStandStorage plugin;
|
||||
private final HashMap<UUID, ArmorStand> openInventories = new HashMap<>();
|
||||
private final boolean isLegacyVersion;
|
||||
|
||||
public ArmorStandListener(ArmorStandStorage plugin) {
|
||||
this.plugin = plugin;
|
||||
// Check if we're running on a legacy version (pre 1.13)
|
||||
isLegacyVersion = !isMethodAvailable("org.bukkit.entity.ArmorStand", "getEquipment");
|
||||
|
||||
// Register all existing armor stands on startup
|
||||
Bukkit.getScheduler().runTaskLater(plugin, () -> {
|
||||
for (org.bukkit.World world : Bukkit.getWorlds()) {
|
||||
for (Entity entity : world.getEntities()) {
|
||||
if (entity instanceof ArmorStand) {
|
||||
plugin.getDatabase().registerArmorStand(entity.getLocation());
|
||||
}
|
||||
}
|
||||
}
|
||||
plugin.getLogger().info("Registered all existing armor stands!");
|
||||
}, 20L);
|
||||
}
|
||||
|
||||
private boolean isMethodAvailable(String className, String methodName) {
|
||||
try {
|
||||
Class<?> clazz = Class.forName(className);
|
||||
clazz.getMethod(methodName);
|
||||
return true;
|
||||
} catch (Exception e) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
private void setArmorStandEquipment(ArmorStand armorStand, ItemStack[] contents) {
|
||||
EntityEquipment equipment = armorStand.getEquipment();
|
||||
if (equipment == null) return;
|
||||
|
||||
equipment.clear();
|
||||
|
||||
try {
|
||||
// Scan entire inventory for armor and items
|
||||
for (int i = 0; i < contents.length; i++) {
|
||||
ItemStack item = contents[i];
|
||||
if (item == null) continue;
|
||||
|
||||
String type = item.getType().name().toUpperCase();
|
||||
|
||||
// Check for armor pieces
|
||||
if (type.endsWith("_HELMET") || type.contains("_HEAD") || type.equals("PLAYER_HEAD") || type.equals("SKULL") || type.equals("SKULL_ITEM")) {
|
||||
equipment.setHelmet(item);
|
||||
}
|
||||
else if (type.endsWith("_CHESTPLATE") || type.equals("ELYTRA")) {
|
||||
equipment.setChestplate(item);
|
||||
}
|
||||
else if (type.endsWith("_LEGGINGS")) {
|
||||
equipment.setLeggings(item);
|
||||
}
|
||||
else if (type.endsWith("_BOOTS")) {
|
||||
equipment.setBoots(item);
|
||||
}
|
||||
// Handle weapons/tools/items for hands
|
||||
else if (i == 4 || (equipment.getHelmet() == null && equipment.getChestplate() == null
|
||||
&& equipment.getLeggings() == null && equipment.getBoots() == null)) {
|
||||
if (isLegacyVersion) {
|
||||
equipment.setItemInHand(item);
|
||||
} else {
|
||||
equipment.setItemInMainHand(item);
|
||||
}
|
||||
}
|
||||
else if (!isLegacyVersion && i == 5) {
|
||||
equipment.setItemInOffHand(item);
|
||||
}
|
||||
}
|
||||
} catch (Exception e) {
|
||||
plugin.getLogger().warning("Error setting armor stand equipment: " + e.getMessage());
|
||||
}
|
||||
}
|
||||
|
||||
private void getArmorStandEquipment(ArmorStand armorStand, Inventory inventory) {
|
||||
EntityEquipment equipment = armorStand.getEquipment();
|
||||
if (equipment == null) return;
|
||||
|
||||
try {
|
||||
inventory.setItem(0, equipment.getHelmet());
|
||||
inventory.setItem(1, equipment.getChestplate());
|
||||
inventory.setItem(2, equipment.getLeggings());
|
||||
inventory.setItem(3, equipment.getBoots());
|
||||
|
||||
// Handle main and off hand differently for legacy versions
|
||||
if (isLegacyVersion) {
|
||||
inventory.setItem(4, equipment.getItemInHand()); // Legacy method
|
||||
} else {
|
||||
inventory.setItem(4, equipment.getItemInMainHand());
|
||||
inventory.setItem(5, equipment.getItemInOffHand());
|
||||
}
|
||||
} catch (Exception e) {
|
||||
plugin.getLogger().warning("Error getting armor stand equipment: " + e.getMessage());
|
||||
}
|
||||
}
|
||||
|
||||
@EventHandler
|
||||
public void onArmorStandPlace(CreatureSpawnEvent event) {
|
||||
if (event.getEntityType() == EntityType.ARMOR_STAND) {
|
||||
plugin.getDatabase().registerArmorStand(event.getLocation());
|
||||
plugin.debug("New armor stand registered at: " + formatLocation(event.getLocation()));
|
||||
}
|
||||
}
|
||||
|
||||
@EventHandler
|
||||
public void onArmorStandRemove(EntityDeathEvent event) {
|
||||
if (event.getEntityType() == EntityType.ARMOR_STAND) {
|
||||
if (plugin.getDatabase().isRegistered(event.getEntity().getLocation())) {
|
||||
plugin.getDatabase().removeArmorStand(event.getEntity().getLocation());
|
||||
plugin.getLogger().info("Registered armor stand removed at: " + formatLocation(event.getEntity().getLocation()));
|
||||
} else {
|
||||
plugin.getLogger().info("Unregistered armor stand removed at: " + formatLocation(event.getEntity().getLocation()));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private String formatLocation(org.bukkit.Location loc) {
|
||||
return String.format("World: %s, X: %.2f, Y: %.2f, Z: %.2f",
|
||||
loc.getWorld().getName(), loc.getX(), loc.getY(), loc.getZ());
|
||||
}
|
||||
|
||||
@EventHandler
|
||||
public void onArmorStandInteract(PlayerInteractAtEntityEvent event) {
|
||||
if (!(event.getRightClicked() instanceof ArmorStand)) {
|
||||
return;
|
||||
}
|
||||
|
||||
Player player = event.getPlayer();
|
||||
ArmorStand armorStand = (ArmorStand) event.getRightClicked();
|
||||
|
||||
// Check if in disabled world
|
||||
if (plugin.isWorldDisabled(player.getWorld().getName())) {
|
||||
player.sendMessage(plugin.getMessage("blocked-world"));
|
||||
return;
|
||||
}
|
||||
|
||||
// Check if player is sneaking
|
||||
if (!player.isSneaking()) {
|
||||
return; // Allow normal armor stand interaction when not sneaking
|
||||
}
|
||||
|
||||
// Check permission
|
||||
if (!player.hasPermission("armorstandstorage.use")) {
|
||||
player.sendMessage(plugin.getMessage("no-permission"));
|
||||
return;
|
||||
}
|
||||
|
||||
event.setCancelled(true);
|
||||
|
||||
// Create inventory with configured size
|
||||
int size = Math.max(27, plugin.getInventoryRows() * 9);
|
||||
Inventory inventory = Bukkit.createInventory(null, size, plugin.getInventoryTitle());
|
||||
|
||||
// First, check if there's saved inventory data
|
||||
ItemStack[] savedItems = plugin.getDatabase().getArmorStandInventory(armorStand.getLocation());
|
||||
|
||||
if (savedItems.length > 0 && hasItems(savedItems)) {
|
||||
// Load saved inventory
|
||||
for (int i = 0; i < Math.min(savedItems.length, size); i++) {
|
||||
inventory.setItem(i, savedItems[i]);
|
||||
}
|
||||
} else {
|
||||
// No saved items, get current equipment
|
||||
EntityEquipment equipment = armorStand.getEquipment();
|
||||
if (equipment != null) {
|
||||
// Store current equipment in first slots
|
||||
inventory.setItem(0, equipment.getHelmet());
|
||||
inventory.setItem(1, equipment.getChestplate());
|
||||
inventory.setItem(2, equipment.getLeggings());
|
||||
inventory.setItem(3, equipment.getBoots());
|
||||
if (isLegacyVersion) {
|
||||
inventory.setItem(4, equipment.getItemInHand());
|
||||
} else {
|
||||
inventory.setItem(4, equipment.getItemInMainHand());
|
||||
inventory.setItem(5, equipment.getItemInOffHand());
|
||||
}
|
||||
|
||||
// Save this initial state to database
|
||||
plugin.getDatabase().saveArmorStand(armorStand.getLocation(), inventory.getContents());
|
||||
}
|
||||
}
|
||||
|
||||
// Open inventory
|
||||
player.openInventory(inventory);
|
||||
openInventories.put(player.getUniqueId(), armorStand);
|
||||
}
|
||||
|
||||
// Helper method to check if an ItemStack array has any non-null items
|
||||
private boolean hasItems(ItemStack[] items) {
|
||||
for (ItemStack item : items) {
|
||||
if (item != null) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
@EventHandler
|
||||
public void onArmorStandManipulate(PlayerArmorStandManipulateEvent event) {
|
||||
ArmorStand armorStand = event.getRightClicked();
|
||||
|
||||
// Don't interfere if player is sneaking (our storage UI handles that)
|
||||
if (event.getPlayer().isSneaking()) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Schedule a task to save the new equipment state after the vanilla interaction
|
||||
Bukkit.getScheduler().runTask(plugin, () -> {
|
||||
if (!plugin.getDatabase().isRegistered(armorStand.getLocation())) {
|
||||
plugin.getDatabase().registerArmorStand(armorStand.getLocation());
|
||||
}
|
||||
|
||||
// Get current equipment and save it
|
||||
EntityEquipment equipment = armorStand.getEquipment();
|
||||
if (equipment != null) {
|
||||
ItemStack[] contents = new ItemStack[27];
|
||||
contents[0] = equipment.getHelmet();
|
||||
contents[1] = equipment.getChestplate();
|
||||
contents[2] = equipment.getLeggings();
|
||||
contents[3] = equipment.getBoots();
|
||||
if (isLegacyVersion) {
|
||||
contents[4] = equipment.getItemInHand();
|
||||
} else {
|
||||
contents[4] = equipment.getItemInMainHand();
|
||||
contents[5] = equipment.getItemInOffHand();
|
||||
}
|
||||
plugin.getDatabase().saveArmorStand(armorStand.getLocation(), contents);
|
||||
plugin.debug("Saved armor stand equipment after manual interaction");
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@EventHandler
|
||||
public void onInventoryClick(InventoryClickEvent event) {
|
||||
Player player = (Player) event.getWhoClicked();
|
||||
ArmorStand armorStand = openInventories.get(player.getUniqueId());
|
||||
|
||||
if (armorStand == null || !event.getView().getTitle().equals(plugin.getInventoryTitle())) {
|
||||
return;
|
||||
}
|
||||
|
||||
plugin.debug("Player " + player.getName() + " clicked in armor stand inventory");
|
||||
|
||||
// Update equipment immediately after click
|
||||
Bukkit.getScheduler().runTask(plugin, () -> {
|
||||
Inventory inv = event.getView().getTopInventory();
|
||||
ItemStack[] contents = inv.getContents();
|
||||
|
||||
// Update armor stand equipment
|
||||
setArmorStandEquipment(armorStand, contents);
|
||||
|
||||
// Save to database
|
||||
plugin.getDatabase().saveArmorStand(armorStand.getLocation(), contents);
|
||||
});
|
||||
}
|
||||
|
||||
@EventHandler
|
||||
public void onInventoryDrag(InventoryDragEvent event) {
|
||||
Player player = (Player) event.getWhoClicked();
|
||||
ArmorStand armorStand = openInventories.get(player.getUniqueId());
|
||||
|
||||
if (armorStand == null || !event.getView().getTitle().equals(plugin.getInventoryTitle())) {
|
||||
return;
|
||||
}
|
||||
|
||||
plugin.debug("Player " + player.getName() + " dragged in armor stand inventory");
|
||||
|
||||
// Update equipment immediately after drag
|
||||
Bukkit.getScheduler().runTask(plugin, () -> {
|
||||
Inventory inv = event.getView().getTopInventory();
|
||||
ItemStack[] contents = inv.getContents();
|
||||
|
||||
// Update armor stand equipment
|
||||
setArmorStandEquipment(armorStand, contents);
|
||||
|
||||
// Save to database
|
||||
plugin.getDatabase().saveArmorStand(armorStand.getLocation(), contents);
|
||||
});
|
||||
}
|
||||
|
||||
@EventHandler
|
||||
public void onInventoryClose(InventoryCloseEvent event) {
|
||||
Player player = (Player) event.getPlayer();
|
||||
ArmorStand armorStand = openInventories.remove(player.getUniqueId());
|
||||
|
||||
if (armorStand == null || !event.getView().getTitle().equals(plugin.getInventoryTitle())) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Save the exact inventory contents, preserving null slots
|
||||
Inventory inv = event.getView().getTopInventory();
|
||||
ItemStack[] contents = new ItemStack[inv.getSize()];
|
||||
|
||||
// Count non-null items while copying contents
|
||||
int itemCount = 0;
|
||||
for (int i = 0; i < inv.getSize(); i++) {
|
||||
contents[i] = inv.getItem(i);
|
||||
if (contents[i] != null) {
|
||||
itemCount++;
|
||||
}
|
||||
}
|
||||
|
||||
// Save to database
|
||||
plugin.getDatabase().saveArmorStand(armorStand.getLocation(), contents);
|
||||
|
||||
// Update armor stand equipment
|
||||
setArmorStandEquipment(armorStand, contents);
|
||||
|
||||
plugin.debug("Saving inventory with " + contents.length + " slots");
|
||||
plugin.debug("Found " + itemCount + " items to save");
|
||||
plugin.debug("Saved armor stand inventory for " + player.getName());
|
||||
}
|
||||
}
|
@ -0,0 +1,79 @@
|
||||
package com.koopacraft.armorstandstorage;
|
||||
|
||||
import org.bukkit.plugin.java.JavaPlugin;
|
||||
import java.io.File;
|
||||
import java.util.List;
|
||||
import java.util.ArrayList;
|
||||
|
||||
public class ArmorStandStorage extends JavaPlugin {
|
||||
private Database database;
|
||||
private List<String> disabledWorlds;
|
||||
private boolean debugMode;
|
||||
|
||||
@Override
|
||||
public void onEnable() {
|
||||
// Create config if it doesn't exist
|
||||
saveDefaultConfig();
|
||||
|
||||
// Load settings
|
||||
disabledWorlds = getConfig().getStringList("disabled-worlds");
|
||||
if (disabledWorlds == null) {
|
||||
disabledWorlds = new ArrayList<>();
|
||||
}
|
||||
debugMode = getConfig().getBoolean("debug", false);
|
||||
|
||||
// Initialize database
|
||||
database = new Database(this, new File(getDataFolder(), "armorstands.db"));
|
||||
|
||||
// Register events
|
||||
getServer().getPluginManager().registerEvents(new ArmorStandListener(this), this);
|
||||
|
||||
// Log startup
|
||||
getLogger().info("ArmorStandStorage has been enabled!");
|
||||
if (!disabledWorlds.isEmpty()) {
|
||||
getLogger().info("Armor stand storage is disabled in: " + String.join(", ", disabledWorlds));
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onDisable() {
|
||||
if (database != null) {
|
||||
database.close();
|
||||
}
|
||||
getLogger().info("ArmorStandStorage has been disabled!");
|
||||
}
|
||||
|
||||
public Database getDatabase() {
|
||||
return database;
|
||||
}
|
||||
|
||||
public String getMessage(String path) {
|
||||
return getConfig().getString("messages." + path, "Message not found: " + path);
|
||||
}
|
||||
|
||||
public String getDisabledWorld() {
|
||||
return getConfig().getString("disabled-world", "world_KC_SPAWN");
|
||||
}
|
||||
|
||||
public String getInventoryTitle() {
|
||||
return getConfig().getString("storage.inventory-title", "Armor Stand Storage");
|
||||
}
|
||||
|
||||
public int getInventoryRows() {
|
||||
return getConfig().getInt("storage.inventory-rows", 3);
|
||||
}
|
||||
|
||||
public boolean isWorldDisabled(String worldName) {
|
||||
return disabledWorlds.contains(worldName);
|
||||
}
|
||||
|
||||
public void debug(String message) {
|
||||
if (debugMode) {
|
||||
getLogger().info("[Debug] " + message);
|
||||
}
|
||||
}
|
||||
|
||||
public boolean isDebugMode() {
|
||||
return debugMode;
|
||||
}
|
||||
}
|
201
src/main/java/com/koopacraft/armorstandstorage/Database.java
Normal file
201
src/main/java/com/koopacraft/armorstandstorage/Database.java
Normal file
@ -0,0 +1,201 @@
|
||||
package com.koopacraft.armorstandstorage;
|
||||
|
||||
import org.bukkit.Location;
|
||||
import org.bukkit.inventory.ItemStack;
|
||||
import org.bukkit.util.io.BukkitObjectInputStream;
|
||||
import org.bukkit.util.io.BukkitObjectOutputStream;
|
||||
|
||||
import java.io.File;
|
||||
import java.io.ByteArrayInputStream;
|
||||
import java.io.ByteArrayOutputStream;
|
||||
import java.sql.*;
|
||||
import java.util.Base64;
|
||||
|
||||
public class Database {
|
||||
private Connection connection;
|
||||
private final ArmorStandStorage plugin;
|
||||
|
||||
public Database(ArmorStandStorage plugin, File file) {
|
||||
this.plugin = plugin;
|
||||
try {
|
||||
Class.forName("org.sqlite.JDBC");
|
||||
connection = DriverManager.getConnection("jdbc:sqlite:" + file.getAbsolutePath());
|
||||
|
||||
// Create tables if they don't exist
|
||||
try (Statement stmt = connection.createStatement()) {
|
||||
stmt.execute("CREATE TABLE IF NOT EXISTS armor_stands (" +
|
||||
"id INTEGER PRIMARY KEY AUTOINCREMENT," +
|
||||
"world TEXT NOT NULL," +
|
||||
"x DOUBLE NOT NULL," +
|
||||
"y DOUBLE NOT NULL," +
|
||||
"z DOUBLE NOT NULL," +
|
||||
"inventory TEXT," +
|
||||
"registered_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP" +
|
||||
")");
|
||||
}
|
||||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
|
||||
public void registerArmorStand(Location location) {
|
||||
try {
|
||||
String sql = "INSERT OR IGNORE INTO armor_stands (world, x, y, z) VALUES (?, ?, ?, ?)";
|
||||
try (PreparedStatement pstmt = connection.prepareStatement(sql)) {
|
||||
pstmt.setString(1, location.getWorld().getName());
|
||||
pstmt.setDouble(2, location.getX());
|
||||
pstmt.setDouble(3, location.getY());
|
||||
pstmt.setDouble(4, location.getZ());
|
||||
pstmt.executeUpdate();
|
||||
}
|
||||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
|
||||
public boolean isRegistered(Location location) {
|
||||
try {
|
||||
String sql = "SELECT id FROM armor_stands WHERE world = ? AND x = ? AND y = ? AND z = ?";
|
||||
try (PreparedStatement pstmt = connection.prepareStatement(sql)) {
|
||||
pstmt.setString(1, location.getWorld().getName());
|
||||
pstmt.setDouble(2, location.getX());
|
||||
pstmt.setDouble(3, location.getY());
|
||||
pstmt.setDouble(4, location.getZ());
|
||||
|
||||
ResultSet rs = pstmt.executeQuery();
|
||||
return rs.next();
|
||||
}
|
||||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
public void removeArmorStand(Location location) {
|
||||
try {
|
||||
String sql = "DELETE FROM armor_stands WHERE world = ? AND x = ? AND y = ? AND z = ?";
|
||||
try (PreparedStatement pstmt = connection.prepareStatement(sql)) {
|
||||
pstmt.setString(1, location.getWorld().getName());
|
||||
pstmt.setDouble(2, location.getX());
|
||||
pstmt.setDouble(3, location.getY());
|
||||
pstmt.setDouble(4, location.getZ());
|
||||
pstmt.executeUpdate();
|
||||
}
|
||||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
|
||||
public void saveArmorStand(Location location, ItemStack[] inventory) {
|
||||
try {
|
||||
String serializedInventory = serializeItems(inventory);
|
||||
plugin.debug("Saving inventory: " + serializedInventory);
|
||||
|
||||
String sql = "UPDATE armor_stands SET inventory = ? WHERE world = ? AND x = ? AND y = ? AND z = ?";
|
||||
try (PreparedStatement pstmt = connection.prepareStatement(sql)) {
|
||||
pstmt.setString(1, serializedInventory);
|
||||
pstmt.setString(2, location.getWorld().getName());
|
||||
pstmt.setDouble(3, location.getX());
|
||||
pstmt.setDouble(4, location.getY());
|
||||
pstmt.setDouble(5, location.getZ());
|
||||
int updated = pstmt.executeUpdate();
|
||||
|
||||
// If no rows were updated, the armor stand might not be registered yet
|
||||
if (updated == 0) {
|
||||
sql = "INSERT INTO armor_stands (world, x, y, z, inventory) VALUES (?, ?, ?, ?, ?)";
|
||||
try (PreparedStatement insertStmt = connection.prepareStatement(sql)) {
|
||||
insertStmt.setString(1, location.getWorld().getName());
|
||||
insertStmt.setDouble(2, location.getX());
|
||||
insertStmt.setDouble(3, location.getY());
|
||||
insertStmt.setDouble(4, location.getZ());
|
||||
insertStmt.setString(5, serializedInventory);
|
||||
insertStmt.executeUpdate();
|
||||
}
|
||||
}
|
||||
}
|
||||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
|
||||
public ItemStack[] getArmorStandInventory(Location location) {
|
||||
try {
|
||||
String sql = "SELECT inventory FROM armor_stands WHERE world = ? AND x = ? AND y = ? AND z = ?";
|
||||
try (PreparedStatement pstmt = connection.prepareStatement(sql)) {
|
||||
pstmt.setString(1, location.getWorld().getName());
|
||||
pstmt.setDouble(2, location.getX());
|
||||
pstmt.setDouble(3, location.getY());
|
||||
pstmt.setDouble(4, location.getZ());
|
||||
|
||||
ResultSet rs = pstmt.executeQuery();
|
||||
if (rs.next()) {
|
||||
String inventoryData = rs.getString("inventory");
|
||||
plugin.debug("Loading inventory: " + inventoryData);
|
||||
if (inventoryData != null && !inventoryData.isEmpty()) {
|
||||
return deserializeItems(inventoryData);
|
||||
}
|
||||
}
|
||||
}
|
||||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
return new ItemStack[27];
|
||||
}
|
||||
|
||||
private String serializeItems(ItemStack[] items) throws Exception {
|
||||
ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
|
||||
BukkitObjectOutputStream dataOutput = new BukkitObjectOutputStream(outputStream);
|
||||
|
||||
// Count non-null items
|
||||
int nonNullCount = 0;
|
||||
for (ItemStack item : items) {
|
||||
if (item != null) nonNullCount++;
|
||||
}
|
||||
dataOutput.writeInt(nonNullCount);
|
||||
|
||||
// Write items with their slot numbers
|
||||
for (int i = 0; i < items.length; i++) {
|
||||
if (items[i] != null) {
|
||||
dataOutput.writeInt(i); // Write the slot number
|
||||
dataOutput.writeObject(items[i]);
|
||||
}
|
||||
}
|
||||
|
||||
dataOutput.close();
|
||||
return Base64.getEncoder().encodeToString(outputStream.toByteArray());
|
||||
}
|
||||
|
||||
private ItemStack[] deserializeItems(String data) throws Exception {
|
||||
if (data == null || data.isEmpty()) {
|
||||
return new ItemStack[27]; // Return empty array with correct size
|
||||
}
|
||||
|
||||
ByteArrayInputStream inputStream = new ByteArrayInputStream(Base64.getDecoder().decode(data));
|
||||
BukkitObjectInputStream dataInput = new BukkitObjectInputStream(inputStream);
|
||||
|
||||
int size = dataInput.readInt();
|
||||
ItemStack[] items = new ItemStack[27]; // Fixed size for chest inventory
|
||||
|
||||
// Read items into their exact positions
|
||||
for (int i = 0; i < size; i++) {
|
||||
int slot = dataInput.readInt(); // Read the slot number
|
||||
ItemStack item = (ItemStack) dataInput.readObject();
|
||||
if (slot >= 0 && slot < items.length) {
|
||||
items[slot] = item;
|
||||
}
|
||||
}
|
||||
|
||||
dataInput.close();
|
||||
return items;
|
||||
}
|
||||
|
||||
public void close() {
|
||||
try {
|
||||
if (connection != null && !connection.isClosed()) {
|
||||
connection.close();
|
||||
}
|
||||
} catch (SQLException e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
}
|
25
src/main/resources/config.yml
Normal file
25
src/main/resources/config.yml
Normal file
@ -0,0 +1,25 @@
|
||||
# ArmorStandStorage Configuration
|
||||
|
||||
# Worlds where armor stand storage is disabled
|
||||
# Leave empty to allow in all worlds
|
||||
# Example:
|
||||
disabled-worlds:
|
||||
#- world_KC_SPAWN
|
||||
#- world_event
|
||||
#- minigames_lobby
|
||||
|
||||
# Debug mode - set to true for detailed logging
|
||||
debug: false
|
||||
|
||||
# Messages
|
||||
messages:
|
||||
no-permission: "§cYou don't have permission to use armor stand storage!"
|
||||
must-sneak: "§cYou must be sneaking to use this!"
|
||||
blocked-world: "§cArmor stand storage is disabled in this world!"
|
||||
|
||||
# Storage settings
|
||||
storage:
|
||||
# Inventory title
|
||||
inventory-title: "Armor Stand Storage"
|
||||
# Number of rows in the storage (must be between 1 and 6)
|
||||
inventory-rows: 3
|
9
src/main/resources/plugin.yml
Normal file
9
src/main/resources/plugin.yml
Normal file
@ -0,0 +1,9 @@
|
||||
name: ArmorStandStorage
|
||||
version: 1.0
|
||||
main: com.koopacraft.armorstandstorage.ArmorStandStorage
|
||||
api-version: 1.13
|
||||
description: Allows players to store items in armor stands
|
||||
permissions:
|
||||
armorstandstorage.use:
|
||||
description: Allows players to use armor stand storage
|
||||
default: op
|
Loading…
x
Reference in New Issue
Block a user