mirror of
https://github.com/KoopaCode/ArmorStand-Storage.git
synced 2025-04-03 22:47:31 +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