Fixed soulbound items being lost when logging off

This commit is contained in:
Jules 2024-06-21 16:29:12 -07:00
parent 56f5e580d0
commit cef7d0163a
3 changed files with 29 additions and 26 deletions

View File

@ -189,7 +189,6 @@ public class MMOItems extends MMOPlugin {
Bukkit.getPluginManager().registerEvents(dropTableManager, this); Bukkit.getPluginManager().registerEvents(dropTableManager, this);
// Load Dist module
// Load MMOCore-Bukkit module // Load MMOCore-Bukkit module
try { try {
Class.forName("net.Indyuce.mmoitems.MMOItemsBukkit").getConstructor(MMOItems.class).newInstance(this); Class.forName("net.Indyuce.mmoitems.MMOItemsBukkit").getConstructor(MMOItems.class).newInstance(this);
@ -268,7 +267,7 @@ public class MMOItems extends MMOPlugin {
playerDataManager.close(); playerDataManager.close();
// Drop abandoned items // Drop abandoned items
DeathItemsHandler.getActive().forEach(info -> info.giveItems(true)); DeathItemsHandler.getActive().forEach(DeathItemsHandler::dropItems);
// Close inventories // Close inventories
for (Player player : Bukkit.getOnlinePlayers()) for (Player player : Bukkit.getOnlinePlayers())

View File

@ -1,14 +1,15 @@
package net.Indyuce.mmoitems.api; package net.Indyuce.mmoitems.api;
import java.util.*;
import net.Indyuce.mmoitems.MMOItems; import net.Indyuce.mmoitems.MMOItems;
import org.bukkit.Bukkit; import org.bukkit.Bukkit;
import org.bukkit.Location;
import org.bukkit.entity.Player; import org.bukkit.entity.Player;
import org.bukkit.inventory.ItemStack; import org.bukkit.inventory.ItemStack;
import org.jetbrains.annotations.NotNull; import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable; import org.jetbrains.annotations.Nullable;
import java.util.*;
public class DeathItemsHandler { public class DeathItemsHandler {
/** /**
@ -21,12 +22,14 @@ public class DeathItemsHandler {
* lost. the plugin saves the last location of the player to drop the items * lost. the plugin saves the last location of the player to drop the items
* when the server shuts down this way they are 'saved' * when the server shuts down this way they are 'saved'
*/ */
private final Player player; private final Location lastLocation;
private final UUID playerId;
/** /**
* Used to store which items must be given back to which player * Used to store which items must be given back to which player
*/ */
private static final Map<UUID, DeathItemsHandler> INFO = new WeakHashMap<>(); private static final Map<UUID, DeathItemsHandler> PLAYER_INFO = new WeakHashMap<>();
/** /**
* Instanced when a player dies if some Soulbound items must be kept in the * Instanced when a player dies if some Soulbound items must be kept in the
@ -37,7 +40,8 @@ public class DeathItemsHandler {
* player respawns again. * player respawns again.
*/ */
public DeathItemsHandler(@NotNull Player player) { public DeathItemsHandler(@NotNull Player player) {
this.player = player; this.playerId = player.getUniqueId();
this.lastLocation = player.getLocation();
} }
public void registerItem(@NotNull ItemStack item) { public void registerItem(@NotNull ItemStack item) {
@ -45,24 +49,23 @@ public class DeathItemsHandler {
} }
public void registerIfNecessary() { public void registerIfNecessary() {
if (!items.isEmpty()) INFO.put(player.getUniqueId(), this); if (!items.isEmpty()) PLAYER_INFO.put(playerId, this);
} }
/** /**
* @param forceDrop Should the items all drop on the ground * Drops items on the ground. Called when the player
* has not respawned yet and items need to be disposed of!
* TODO save it inside of configuration files and stop using this method. Give items back when joining
*/ */
public void giveItems(boolean forceDrop) { public void dropItems() {
for (ItemStack drop : items)
lastLocation.getWorld().dropItem(lastLocation, drop);
}
// Drop all items on the ground public void giveItems(@NotNull Player player) {
if (forceDrop) for (ItemStack drop : items) final ItemStack[] toArray = this.items.toArray(new ItemStack[0]);
for (ItemStack drop : player.getInventory().addItem(toArray).values())
player.getWorld().dropItem(player.getLocation(), drop); player.getWorld().dropItem(player.getLocation(), drop);
// First try to add them to inventory
else {
final ItemStack[] toArray = this.items.toArray(new ItemStack[0]);
for (ItemStack drop : player.getInventory().addItem(toArray).values())
player.getWorld().dropItem(player.getLocation(), drop);
}
} }
/** /**
@ -73,15 +76,16 @@ public class DeathItemsHandler {
* @param player Target player respawning * @param player Target player respawning
*/ */
public static void readAndRemove(@NotNull Player player) { public static void readAndRemove(@NotNull Player player) {
final @Nullable DeathItemsHandler handler = INFO.remove(player.getUniqueId()); final @Nullable DeathItemsHandler handler = PLAYER_INFO.remove(player.getUniqueId());
if (handler != null) Bukkit.getScheduler().runTaskLater(MMOItems.plugin, () -> handler.giveItems(false), 10); if (handler != null) Bukkit.getScheduler().runTaskLater(MMOItems.plugin, () -> handler.giveItems(player), 10);
} }
/** /**
* @return Soulbound info of players who have not clicked the respawn button * @return Soulbound info of players who have not clicked the respawn button
* and yet have items cached in server RAM * and yet have items cached in server RAM
*/ */
@NotNull
public static Collection<DeathItemsHandler> getActive() { public static Collection<DeathItemsHandler> getActive() {
return INFO.values(); return PLAYER_INFO.values();
} }
} }

View File

@ -3,7 +3,7 @@ config-version: 8
# Notifies players with the 'mmoitems.update-notify' perm node when # Notifies players with the 'mmoitems.update-notify' perm node when
# they join the server if a new update is available for download. # they join the server if a new update is available for download.
# Requires /reload when changed. # Changes apply on server restart.
update-notify: true update-notify: true
# Enable/disable the plugin iterating over the whole player inventory # Enable/disable the plugin iterating over the whole player inventory
@ -68,7 +68,7 @@ default:
range: 16 range: 16
recoil: 0.1 recoil: 0.1
# Restart server when changing this option # Changes apply on server restart.
dropped-items: dropped-items:
# Items glow based on their tier # Items glow based on their tier
@ -375,7 +375,7 @@ item-revision:
# Offset is the distance traveled on X and Y coordinates # Offset is the distance traveled on X and Y coordinates
# Height is the Y velocity coordinate. Lootsplosions # Height is the Y velocity coordinate. Lootsplosions
# only trigger with MythicMobs monsters. # only trigger with MythicMobs monsters.
# Requires a SERVER reload when changed. # Changes apply on server restart.
lootsplosion: lootsplosion:
enabled: true enabled: true
color: true color: true