Fix clearing specific items in clearinventory (#3263)

Fixes #2986
Closes #3050
Closes #3191

Refactors the clearinventory command code in such a way that fixes the problem of not being able to clear all of a specific item on newer server versions. Also re-adds data value support for older server versions that support it, which seemed to have been inadvertently removed in 79bc340.

Checked against 1.8.8, 1.12.2, 1.13.2, 1.15.2.
This commit is contained in:
pop4959 2020-05-17 13:10:12 -07:00 committed by GitHub
parent b9804d8980
commit 5b9b76d5e7
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23

View file

@ -5,6 +5,7 @@ import com.earth2me.essentials.User;
import com.earth2me.essentials.craftbukkit.InventoryWorkaround; import com.earth2me.essentials.craftbukkit.InventoryWorkaround;
import com.earth2me.essentials.utils.NumberUtil; import com.earth2me.essentials.utils.NumberUtil;
import com.earth2me.essentials.utils.StringUtil; import com.earth2me.essentials.utils.StringUtil;
import com.earth2me.essentials.utils.VersionUtil;
import org.bukkit.Material; import org.bukkit.Material;
import org.bukkit.Server; import org.bukkit.Server;
import org.bukkit.entity.Player; import org.bukkit.entity.Player;
@ -79,39 +80,65 @@ public class Commandclearinventory extends EssentialsCommand {
} }
} }
protected void clearHandler(CommandSource sender, Player player, String[] args, int offset, boolean showExtended) throws Exception { private static class Item {
short data = -1; private Material material;
int type = -1; private short data;
public Item(Material material, short data) {
this.material = material;
this.data = data;
}
public Material getMaterial() {
return material;
}
public short getData() {
return data;
}
}
private enum ClearHandlerType {
ALL_EXCEPT_ARMOR, ALL_INCLUDING_ARMOR, SPECIFIC_ITEM
}
protected void clearHandler(CommandSource sender, Player player, String[] args, int offset, boolean showExtended) {
ClearHandlerType type = ClearHandlerType.ALL_EXCEPT_ARMOR;
final Set<Item> items = new HashSet<>();
int amount = -1; int amount = -1;
final Set<Material> mats = new HashSet<>();
if (args.length > (offset + 1) && NumberUtil.isInt(args[(offset + 1)])) { if (args.length > (offset + 1) && NumberUtil.isInt(args[(offset + 1)])) {
amount = Integer.parseInt(args[(offset + 1)]); amount = Integer.parseInt(args[(offset + 1)]);
} }
if (args.length > offset) { if (args.length > offset) {
if (args[offset].equalsIgnoreCase("**")) { if (args[offset].equalsIgnoreCase("**")) {
type = -2; type = ClearHandlerType.ALL_INCLUDING_ARMOR;
} else if (!args[offset].equalsIgnoreCase("*")) { } else if (!args[offset].equalsIgnoreCase("*")) {
final String[] split = args[offset].split(","); final String[] split = args[offset].split(",");
for (String item : split) {
for (String name : split) { final String[] itemParts = item.split(":");
short data;
try { try {
mats.add(ess.getItemDb().get(name).getType()); data = Short.parseShort(itemParts[1]);
} catch (Exception e) {
data = 0;
}
try {
items.add(new Item(ess.getItemDb().get(itemParts[0]).getType(), data));
} catch (Exception ignored) {} } catch (Exception ignored) {}
} }
type = ClearHandlerType.SPECIFIC_ITEM;
type = 1;
} }
} }
if (type == -1) // type -1 represents wildcard or all items if (type == ClearHandlerType.ALL_EXCEPT_ARMOR)
{ {
if (showExtended) { if (showExtended) {
sender.sendMessage(tl("inventoryClearingAllItems", player.getDisplayName())); sender.sendMessage(tl("inventoryClearingAllItems", player.getDisplayName()));
} }
InventoryWorkaround.clearInventoryNoArmor(player.getInventory()); InventoryWorkaround.clearInventoryNoArmor(player.getInventory());
InventoryWorkaround.setItemInOffHand(player, null); InventoryWorkaround.setItemInOffHand(player, null);
} else if (type == -2) // type -2 represents double wildcard or all items and armor } else if (type == ClearHandlerType.ALL_INCLUDING_ARMOR)
{ {
if (showExtended) { if (showExtended) {
sender.sendMessage(tl("inventoryClearingAllArmor", player.getDisplayName())); sender.sendMessage(tl("inventoryClearingAllArmor", player.getDisplayName()));
@ -120,20 +147,21 @@ public class Commandclearinventory extends EssentialsCommand {
InventoryWorkaround.setItemInOffHand(player, null); InventoryWorkaround.setItemInOffHand(player, null);
player.getInventory().setArmorContents(null); player.getInventory().setArmorContents(null);
} else { } else {
for (Material mat : mats) { for (Item item : items) {
ItemStack stack = new ItemStack(item.getMaterial());
if (VersionUtil.getServerBukkitVersion().isLowerThan(VersionUtil.v1_13_0_R01)) {
stack.setDurability(item.getData());
}
if (amount == -1) // amount -1 means all items will be cleared if (amount == -1) // amount -1 means all items will be cleared
{ {
ItemStack stack = new ItemStack(mat, BASE_AMOUNT, data); stack.setAmount(BASE_AMOUNT);
ItemStack removedStack = player.getInventory().removeItem(stack).get(0); ItemStack removedStack = player.getInventory().removeItem(stack).get(0);
final int removedAmount = (BASE_AMOUNT - removedStack.getAmount()); final int removedAmount = (BASE_AMOUNT - removedStack.getAmount());
if (removedAmount > 0 || showExtended) { if (removedAmount > 0 || showExtended) {
sender.sendMessage(tl("inventoryClearingStack", removedAmount, stack.getType().toString().toLowerCase(Locale.ENGLISH), player.getDisplayName())); sender.sendMessage(tl("inventoryClearingStack", removedAmount, stack.getType().toString().toLowerCase(Locale.ENGLISH), player.getDisplayName()));
} }
} else { } else {
if (amount < 0) { stack.setAmount(amount < 0 ? 1 : amount);
amount = 1;
}
ItemStack stack = new ItemStack(mat, amount);
if (player.getInventory().containsAtLeast(stack, amount)) { if (player.getInventory().containsAtLeast(stack, amount)) {
sender.sendMessage(tl("inventoryClearingStack", amount, stack.getType().toString().toLowerCase(Locale.ENGLISH), player.getDisplayName())); sender.sendMessage(tl("inventoryClearingStack", amount, stack.getType().toString().toLowerCase(Locale.ENGLISH), player.getDisplayName()));
player.getInventory().removeItem(stack); player.getInventory().removeItem(stack);