mirror of
https://github.com/Minestom/Minestom.git
synced 2025-01-24 00:51:34 +01:00
commit
189b8fd552
@ -91,7 +91,7 @@ public final class UpdateManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Flush all waiting packets
|
// Flush all waiting packets
|
||||||
AsyncUtils.runAsync(() -> connectionManager.getOnlinePlayers().stream()
|
AsyncUtils.runAsync(() -> connectionManager.getOnlinePlayers().parallelStream()
|
||||||
.filter(player -> player.getPlayerConnection() instanceof NettyPlayerConnection)
|
.filter(player -> player.getPlayerConnection() instanceof NettyPlayerConnection)
|
||||||
.map(player -> (NettyPlayerConnection) player.getPlayerConnection())
|
.map(player -> (NettyPlayerConnection) player.getPlayerConnection())
|
||||||
.forEach(NettyPlayerConnection::flush));
|
.forEach(NettyPlayerConnection::flush));
|
||||||
|
@ -60,7 +60,7 @@ public class ArgumentWord extends Argument<String> {
|
|||||||
// Check restrictions (acting as literal)
|
// Check restrictions (acting as literal)
|
||||||
if (hasRestrictions()) {
|
if (hasRestrictions()) {
|
||||||
for (String r : restrictions) {
|
for (String r : restrictions) {
|
||||||
if (input.equalsIgnoreCase(r))
|
if (input.equals(r))
|
||||||
return input;
|
return input;
|
||||||
}
|
}
|
||||||
throw new ArgumentSyntaxException("Word needs to be in the restriction list", input, RESTRICTION_ERROR);
|
throw new ArgumentSyntaxException("Word needs to be in the restriction list", input, RESTRICTION_ERROR);
|
||||||
|
@ -55,8 +55,7 @@ public abstract class AbstractInventory implements InventoryClickHandler, DataCo
|
|||||||
public synchronized <T> @NotNull T processItemStack(@NotNull ItemStack itemStack,
|
public synchronized <T> @NotNull T processItemStack(@NotNull ItemStack itemStack,
|
||||||
@NotNull TransactionType type,
|
@NotNull TransactionType type,
|
||||||
@NotNull TransactionOption<T> option) {
|
@NotNull TransactionOption<T> option) {
|
||||||
var pair = type.process(this, itemStack);
|
return option.fill(type, this, itemStack);
|
||||||
return option.fill(this, pair.left(), pair.right());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public synchronized <T> @NotNull List<@NotNull T> processItemStacks(@NotNull List<@NotNull ItemStack> itemStacks,
|
public synchronized <T> @NotNull List<@NotNull T> processItemStacks(@NotNull List<@NotNull ItemStack> itemStacks,
|
||||||
|
@ -4,7 +4,6 @@ import net.kyori.adventure.text.Component;
|
|||||||
import net.minestom.server.Viewable;
|
import net.minestom.server.Viewable;
|
||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.inventory.click.ClickType;
|
import net.minestom.server.inventory.click.ClickType;
|
||||||
import net.minestom.server.inventory.click.InventoryClickLoopHandler;
|
|
||||||
import net.minestom.server.inventory.click.InventoryClickResult;
|
import net.minestom.server.inventory.click.InventoryClickResult;
|
||||||
import net.minestom.server.item.ItemStack;
|
import net.minestom.server.item.ItemStack;
|
||||||
import net.minestom.server.network.packet.server.play.OpenWindowPacket;
|
import net.minestom.server.network.packet.server.play.OpenWindowPacket;
|
||||||
@ -338,51 +337,21 @@ public class Inventory extends AbstractInventory implements Viewable {
|
|||||||
final ItemStack clicked = isInWindow ? getItemStack(slot) : playerInventory.getItemStack(clickSlot);
|
final ItemStack clicked = isInWindow ? getItemStack(slot) : playerInventory.getItemStack(clickSlot);
|
||||||
final ItemStack cursor = getCursorItem(player); // Isn't used in the algorithm
|
final ItemStack cursor = getCursorItem(player); // Isn't used in the algorithm
|
||||||
|
|
||||||
|
final InventoryClickResult clickResult = clickProcessor.shiftClick(
|
||||||
final InventoryClickResult clickResult;
|
isInWindow ? playerInventory : this,
|
||||||
|
isInWindow ? this : null,
|
||||||
if (isInWindow) {
|
player, slot, clicked, cursor);
|
||||||
clickResult = clickProcessor.shiftClick(this, player, slot, clicked, cursor,
|
|
||||||
// Player inventory loop
|
|
||||||
new InventoryClickLoopHandler(0, PlayerInventory.INVENTORY_SIZE, 1,
|
|
||||||
PlayerInventoryUtils::convertToPacketSlot,
|
|
||||||
index -> isClickInWindow(index) ?
|
|
||||||
getItemStack(index) :
|
|
||||||
playerInventory.getItemStack(PlayerInventoryUtils.convertSlot(index, offset)),
|
|
||||||
(index, itemStack) -> {
|
|
||||||
if (isClickInWindow(index)) {
|
|
||||||
setItemStack(index, itemStack);
|
|
||||||
} else {
|
|
||||||
playerInventory.setItemStack(PlayerInventoryUtils.convertSlot(index, offset), itemStack);
|
|
||||||
}
|
|
||||||
}));
|
|
||||||
} else {
|
|
||||||
clickResult = clickProcessor.shiftClick(null, player, slot, clicked, cursor,
|
|
||||||
// Window loop
|
|
||||||
new InventoryClickLoopHandler(0, getSize(), 1,
|
|
||||||
i -> i,
|
|
||||||
index -> isClickInWindow(index) ?
|
|
||||||
getItemStack(index) :
|
|
||||||
playerInventory.getItemStack(PlayerInventoryUtils.convertSlot(index, offset)),
|
|
||||||
(index, itemStack) -> {
|
|
||||||
if (isClickInWindow(index)) {
|
|
||||||
setItemStack(index, itemStack);
|
|
||||||
} else {
|
|
||||||
playerInventory.setItemStack(PlayerInventoryUtils.convertSlot(index, offset), itemStack);
|
|
||||||
}
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (clickResult == null)
|
if (clickResult == null)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (clickResult.doRefresh()) {
|
if (isInWindow) {
|
||||||
updateFromClick(clickResult, player);
|
setItemStack(slot, clickResult.getClicked());
|
||||||
|
} else {
|
||||||
|
playerInventory.setItemStack(clickSlot, clickResult.getClicked());
|
||||||
}
|
}
|
||||||
|
|
||||||
refreshPlayerCursorItem(player, clickResult.getCursor());
|
refreshPlayerCursorItem(player, clickResult.getCursor());
|
||||||
playerInventory.update();
|
|
||||||
update();
|
|
||||||
|
|
||||||
return !clickResult.isCancel();
|
return !clickResult.isCancel();
|
||||||
}
|
}
|
||||||
@ -498,17 +467,8 @@ public class Inventory extends AbstractInventory implements Viewable {
|
|||||||
final ItemStack cursor = getCursorItem(player);
|
final ItemStack cursor = getCursorItem(player);
|
||||||
final boolean isInWindow = isClickInWindow(slot);
|
final boolean isInWindow = isClickInWindow(slot);
|
||||||
|
|
||||||
final InventoryClickResult clickResult = clickProcessor.doubleClick(isInWindow ? this : null, player, slot, cursor,
|
final InventoryClickResult clickResult = clickProcessor.doubleClick(isInWindow ? this : playerInventory,
|
||||||
// Start by looping through the opened inventory
|
this, player, slot, cursor);
|
||||||
new InventoryClickLoopHandler(0, getSize(), 1,
|
|
||||||
i -> i,
|
|
||||||
this::getItemStack,
|
|
||||||
this::setItemStack),
|
|
||||||
// Looping through player inventory
|
|
||||||
new InventoryClickLoopHandler(0, PlayerInventory.INVENTORY_SIZE, 1,
|
|
||||||
PlayerInventoryUtils::convertToPacketSlot,
|
|
||||||
index -> playerInventory.getItemStack(index, PlayerInventoryUtils.OFFSET),
|
|
||||||
(index, itemStack) -> playerInventory.setItemStack(index, PlayerInventoryUtils.OFFSET, itemStack)));
|
|
||||||
|
|
||||||
if (clickResult == null)
|
if (clickResult == null)
|
||||||
return false;
|
return false;
|
||||||
|
@ -3,7 +3,6 @@ package net.minestom.server.inventory;
|
|||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.event.item.ArmorEquipEvent;
|
import net.minestom.server.event.item.ArmorEquipEvent;
|
||||||
import net.minestom.server.inventory.click.ClickType;
|
import net.minestom.server.inventory.click.ClickType;
|
||||||
import net.minestom.server.inventory.click.InventoryClickLoopHandler;
|
|
||||||
import net.minestom.server.inventory.click.InventoryClickResult;
|
import net.minestom.server.inventory.click.InventoryClickResult;
|
||||||
import net.minestom.server.inventory.condition.InventoryCondition;
|
import net.minestom.server.inventory.condition.InventoryCondition;
|
||||||
import net.minestom.server.item.ItemStack;
|
import net.minestom.server.item.ItemStack;
|
||||||
@ -339,26 +338,17 @@ public class PlayerInventory extends AbstractInventory implements EquipmentHandl
|
|||||||
public boolean shiftClick(@NotNull Player player, int slot) {
|
public boolean shiftClick(@NotNull Player player, int slot) {
|
||||||
final ItemStack cursor = getCursorItem();
|
final ItemStack cursor = getCursorItem();
|
||||||
final ItemStack clicked = getItemStack(slot, OFFSET);
|
final ItemStack clicked = getItemStack(slot, OFFSET);
|
||||||
|
final boolean hotBarClick = convertSlot(slot, OFFSET) < 9;
|
||||||
final boolean hotBarClick = convertToPacketSlot(slot) < 9;
|
final int start = hotBarClick ? 9 : 0;
|
||||||
final InventoryClickResult clickResult = clickProcessor.shiftClick(null, player, slot, clicked, cursor,
|
final int end = hotBarClick ? getSize() - 9 : 8;
|
||||||
new InventoryClickLoopHandler(0, itemStacks.length, 1,
|
final InventoryClickResult clickResult = clickProcessor.shiftClick(this,
|
||||||
i -> {
|
start, end, 1,
|
||||||
if (hotBarClick) {
|
player, slot, clicked, cursor);
|
||||||
return i < 9 ? i + 9 : i - 9;
|
|
||||||
} else {
|
|
||||||
return convertPlayerInventorySlot(i, OFFSET);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
index -> getItemStack(index, OFFSET),
|
|
||||||
(index, itemStack) -> setItemStack(index, OFFSET, itemStack)));
|
|
||||||
|
|
||||||
if (clickResult == null)
|
if (clickResult == null)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (clickResult.doRefresh())
|
setItemStack(slot, OFFSET, clickResult.getClicked());
|
||||||
update();
|
|
||||||
|
|
||||||
setCursorItem(clickResult.getCursor());
|
setCursorItem(clickResult.getCursor());
|
||||||
|
|
||||||
return !clickResult.isCancel();
|
return !clickResult.isCancel();
|
||||||
@ -415,12 +405,7 @@ public class PlayerInventory extends AbstractInventory implements EquipmentHandl
|
|||||||
@Override
|
@Override
|
||||||
public boolean doubleClick(@NotNull Player player, int slot) {
|
public boolean doubleClick(@NotNull Player player, int slot) {
|
||||||
final ItemStack cursor = getCursorItem();
|
final ItemStack cursor = getCursorItem();
|
||||||
|
final InventoryClickResult clickResult = clickProcessor.doubleClick(this, null, player, slot, cursor);
|
||||||
final InventoryClickResult clickResult = clickProcessor.doubleClick(null, player, slot, cursor,
|
|
||||||
new InventoryClickLoopHandler(0, itemStacks.length, 1,
|
|
||||||
i -> i < 9 ? i + 9 : i - 9,
|
|
||||||
index -> itemStacks[index],
|
|
||||||
this::setItemStack));
|
|
||||||
|
|
||||||
if (clickResult == null)
|
if (clickResult == null)
|
||||||
return false;
|
return false;
|
||||||
@ -428,6 +413,7 @@ public class PlayerInventory extends AbstractInventory implements EquipmentHandl
|
|||||||
if (clickResult.doRefresh())
|
if (clickResult.doRefresh())
|
||||||
update();
|
update();
|
||||||
|
|
||||||
|
setItemStack(slot, OFFSET, clickResult.getClicked());
|
||||||
setCursorItem(clickResult.getCursor());
|
setCursorItem(clickResult.getCursor());
|
||||||
|
|
||||||
return !clickResult.isCancel();
|
return !clickResult.isCancel();
|
||||||
|
@ -44,4 +44,11 @@ public interface TransactionOption<T> {
|
|||||||
@NotNull T fill(@NotNull AbstractInventory inventory,
|
@NotNull T fill(@NotNull AbstractInventory inventory,
|
||||||
@NotNull ItemStack result,
|
@NotNull ItemStack result,
|
||||||
@NotNull Map<@NotNull Integer, @NotNull ItemStack> itemChangesMap);
|
@NotNull Map<@NotNull Integer, @NotNull ItemStack> itemChangesMap);
|
||||||
|
|
||||||
|
default @NotNull T fill(@NotNull TransactionType type,
|
||||||
|
@NotNull AbstractInventory inventory,
|
||||||
|
@NotNull ItemStack itemStack) {
|
||||||
|
var pair = type.process(inventory, itemStack);
|
||||||
|
return fill(inventory, pair.left(), pair.right());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -19,13 +19,13 @@ public interface TransactionType {
|
|||||||
* Adds an item to the inventory.
|
* Adds an item to the inventory.
|
||||||
* Can either take an air slot or be stacked.
|
* Can either take an air slot or be stacked.
|
||||||
*/
|
*/
|
||||||
TransactionType ADD = (inventory, itemStack) -> {
|
TransactionType ADD = (inventory, itemStack, slotPredicate, start, end, step) -> {
|
||||||
Int2ObjectMap<ItemStack> itemChangesMap = new Int2ObjectOpenHashMap<>();
|
Int2ObjectMap<ItemStack> itemChangesMap = new Int2ObjectOpenHashMap<>();
|
||||||
|
|
||||||
final StackingRule stackingRule = itemStack.getStackingRule();
|
final StackingRule stackingRule = itemStack.getStackingRule();
|
||||||
|
|
||||||
// Check filled slot (not air)
|
// Check filled slot (not air)
|
||||||
for (int i = 0; i < inventory.getInnerSize(); i++) {
|
for (int i = start; i < end; i += step) {
|
||||||
ItemStack inventoryItem = inventory.getItemStack(i);
|
ItemStack inventoryItem = inventory.getItemStack(i);
|
||||||
if (inventoryItem.isAir()) {
|
if (inventoryItem.isAir()) {
|
||||||
continue;
|
continue;
|
||||||
@ -34,6 +34,12 @@ public interface TransactionType {
|
|||||||
final int itemAmount = stackingRule.getAmount(inventoryItem);
|
final int itemAmount = stackingRule.getAmount(inventoryItem);
|
||||||
if (itemAmount == stackingRule.getMaxSize())
|
if (itemAmount == stackingRule.getMaxSize())
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
if (!slotPredicate.test(i, inventoryItem)) {
|
||||||
|
// Cancelled transaction
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
final int itemStackAmount = stackingRule.getAmount(itemStack);
|
final int itemStackAmount = stackingRule.getAmount(itemStack);
|
||||||
final int totalAmount = itemStackAmount + itemAmount;
|
final int totalAmount = itemStackAmount + itemAmount;
|
||||||
if (!stackingRule.canApply(itemStack, totalAmount)) {
|
if (!stackingRule.canApply(itemStack, totalAmount)) {
|
||||||
@ -50,11 +56,17 @@ public interface TransactionType {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Check air slot to fill
|
// Check air slot to fill
|
||||||
for (int i = 0; i < inventory.getInnerSize(); i++) {
|
for (int i = start; i < end; i += step) {
|
||||||
ItemStack inventoryItem = inventory.getItemStack(i);
|
ItemStack inventoryItem = inventory.getItemStack(i);
|
||||||
if (!inventoryItem.isAir()) {
|
if (!inventoryItem.isAir()) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!slotPredicate.test(i, inventoryItem)) {
|
||||||
|
// Cancelled transaction
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
// Fill the slot
|
// Fill the slot
|
||||||
itemChangesMap.put(i, itemStack);
|
itemChangesMap.put(i, itemStack);
|
||||||
itemStack = stackingRule.apply(itemStack, 0);
|
itemStack = stackingRule.apply(itemStack, 0);
|
||||||
@ -68,15 +80,20 @@ public interface TransactionType {
|
|||||||
* Takes an item from the inventory.
|
* Takes an item from the inventory.
|
||||||
* Can either transform items to air or reduce their amount.
|
* Can either transform items to air or reduce their amount.
|
||||||
*/
|
*/
|
||||||
TransactionType TAKE = (inventory, itemStack) -> {
|
TransactionType TAKE = (inventory, itemStack, slotPredicate, start, end, step) -> {
|
||||||
Int2ObjectMap<ItemStack> itemChangesMap = new Int2ObjectOpenHashMap<>();
|
Int2ObjectMap<ItemStack> itemChangesMap = new Int2ObjectOpenHashMap<>();
|
||||||
final StackingRule stackingRule = itemStack.getStackingRule();
|
final StackingRule stackingRule = itemStack.getStackingRule();
|
||||||
for (int i = 0; i < inventory.getInnerSize(); i++) {
|
for (int i = start; i < end; i += step) {
|
||||||
ItemStack inventoryItem = inventory.getItemStack(i);
|
ItemStack inventoryItem = inventory.getItemStack(i);
|
||||||
if (inventoryItem.isAir()) {
|
if (inventoryItem.isAir()) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
if (stackingRule.canBeStacked(itemStack, inventoryItem)) {
|
if (stackingRule.canBeStacked(itemStack, inventoryItem)) {
|
||||||
|
if (!slotPredicate.test(i, inventoryItem)) {
|
||||||
|
// Cancelled transaction
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
final int itemAmount = stackingRule.getAmount(inventoryItem);
|
final int itemAmount = stackingRule.getAmount(inventoryItem);
|
||||||
final int itemStackAmount = stackingRule.getAmount(itemStack);
|
final int itemStackAmount = stackingRule.getAmount(itemStack);
|
||||||
if (itemStackAmount < itemAmount) {
|
if (itemStackAmount < itemAmount) {
|
||||||
@ -97,6 +114,23 @@ public interface TransactionType {
|
|||||||
};
|
};
|
||||||
|
|
||||||
@NotNull Pair<ItemStack, Map<Integer, ItemStack>> process(@NotNull AbstractInventory inventory,
|
@NotNull Pair<ItemStack, Map<Integer, ItemStack>> process(@NotNull AbstractInventory inventory,
|
||||||
@NotNull ItemStack itemStack);
|
@NotNull ItemStack itemStack,
|
||||||
|
@NotNull SlotPredicate slotPredicate,
|
||||||
|
int start, int end, int step);
|
||||||
|
|
||||||
|
default @NotNull Pair<ItemStack, Map<Integer, ItemStack>> process(@NotNull AbstractInventory inventory,
|
||||||
|
@NotNull ItemStack itemStack,
|
||||||
|
@NotNull SlotPredicate slotPredicate) {
|
||||||
|
return process(inventory, itemStack, slotPredicate, 0, inventory.getInnerSize(), 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
default @NotNull Pair<ItemStack, Map<Integer, ItemStack>> process(@NotNull AbstractInventory inventory,
|
||||||
|
@NotNull ItemStack itemStack) {
|
||||||
|
return process(inventory, itemStack, (slot, itemStack1) -> true);
|
||||||
|
}
|
||||||
|
|
||||||
|
@FunctionalInterface
|
||||||
|
interface SlotPredicate {
|
||||||
|
boolean test(int slot, @NotNull ItemStack itemStack);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,53 +0,0 @@
|
|||||||
package net.minestom.server.inventory.click;
|
|
||||||
|
|
||||||
import it.unimi.dsi.fastutil.ints.Int2IntFunction;
|
|
||||||
import it.unimi.dsi.fastutil.ints.Int2ObjectFunction;
|
|
||||||
import net.minestom.server.item.ItemStack;
|
|
||||||
|
|
||||||
import java.util.function.BiConsumer;
|
|
||||||
|
|
||||||
public class InventoryClickLoopHandler {
|
|
||||||
|
|
||||||
private final int start;
|
|
||||||
private final int end;
|
|
||||||
private final int step;
|
|
||||||
private final Int2IntFunction indexModifier;
|
|
||||||
private final Int2ObjectFunction<ItemStack> itemGetter;
|
|
||||||
private final BiConsumer<Integer, ItemStack> itemSetter;
|
|
||||||
|
|
||||||
public InventoryClickLoopHandler(int start, int end, int step,
|
|
||||||
Int2IntFunction indexModifier,
|
|
||||||
Int2ObjectFunction<ItemStack> itemGetter,
|
|
||||||
BiConsumer<Integer, ItemStack> itemSetter) {
|
|
||||||
this.start = start;
|
|
||||||
this.end = end;
|
|
||||||
this.step = step;
|
|
||||||
this.indexModifier = indexModifier;
|
|
||||||
this.itemGetter = itemGetter;
|
|
||||||
this.itemSetter = itemSetter;
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getStart() {
|
|
||||||
return start;
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getEnd() {
|
|
||||||
return end;
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getStep() {
|
|
||||||
return step;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Int2IntFunction getIndexModifier() {
|
|
||||||
return indexModifier;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Int2ObjectFunction<ItemStack> getItemGetter() {
|
|
||||||
return itemGetter;
|
|
||||||
}
|
|
||||||
|
|
||||||
public BiConsumer<Integer, ItemStack> getItemSetter() {
|
|
||||||
return itemSetter;
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,31 +1,37 @@
|
|||||||
package net.minestom.server.inventory.click;
|
package net.minestom.server.inventory.click;
|
||||||
|
|
||||||
import it.unimi.dsi.fastutil.ints.Int2IntFunction;
|
|
||||||
import it.unimi.dsi.fastutil.ints.Int2ObjectFunction;
|
import it.unimi.dsi.fastutil.ints.Int2ObjectFunction;
|
||||||
import it.unimi.dsi.fastutil.ints.IntOpenHashSet;
|
import it.unimi.dsi.fastutil.ints.IntOpenHashSet;
|
||||||
import it.unimi.dsi.fastutil.ints.IntSet;
|
import it.unimi.dsi.fastutil.ints.IntSet;
|
||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.event.inventory.InventoryClickEvent;
|
import net.minestom.server.event.inventory.InventoryClickEvent;
|
||||||
import net.minestom.server.event.inventory.InventoryPreClickEvent;
|
import net.minestom.server.event.inventory.InventoryPreClickEvent;
|
||||||
|
import net.minestom.server.inventory.AbstractInventory;
|
||||||
import net.minestom.server.inventory.Inventory;
|
import net.minestom.server.inventory.Inventory;
|
||||||
|
import net.minestom.server.inventory.TransactionOption;
|
||||||
|
import net.minestom.server.inventory.TransactionType;
|
||||||
import net.minestom.server.inventory.condition.InventoryCondition;
|
import net.minestom.server.inventory.condition.InventoryCondition;
|
||||||
import net.minestom.server.inventory.condition.InventoryConditionResult;
|
import net.minestom.server.inventory.condition.InventoryConditionResult;
|
||||||
import net.minestom.server.item.ItemStack;
|
import net.minestom.server.item.ItemStack;
|
||||||
import net.minestom.server.item.StackingRule;
|
import net.minestom.server.item.StackingRule;
|
||||||
import net.minestom.server.utils.inventory.PlayerInventoryUtils;
|
import net.minestom.server.utils.inventory.PlayerInventoryUtils;
|
||||||
|
import org.jetbrains.annotations.ApiStatus;
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
import org.jetbrains.annotations.Nullable;
|
import org.jetbrains.annotations.Nullable;
|
||||||
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
import java.util.Objects;
|
||||||
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
import java.util.function.BiConsumer;
|
import java.util.function.BiConsumer;
|
||||||
|
import java.util.function.BiFunction;
|
||||||
|
|
||||||
|
@ApiStatus.Internal
|
||||||
public class InventoryClickProcessor {
|
public class InventoryClickProcessor {
|
||||||
|
|
||||||
// Dragging maps
|
// Dragging maps
|
||||||
private final Map<Player, IntSet> leftDraggingMap = new HashMap<>();
|
private final Map<Player, IntSet> leftDraggingMap = new ConcurrentHashMap<>();
|
||||||
private final Map<Player, IntSet> rightDraggingMap = new HashMap<>();
|
private final Map<Player, IntSet> rightDraggingMap = new ConcurrentHashMap<>();
|
||||||
|
|
||||||
@NotNull
|
@NotNull
|
||||||
public InventoryClickResult leftClick(@Nullable Inventory inventory, @NotNull Player player, int slot,
|
public InventoryClickResult leftClick(@Nullable Inventory inventory, @NotNull Player player, int slot,
|
||||||
@ -167,9 +173,8 @@ public class InventoryClickProcessor {
|
|||||||
return clickResult;
|
return clickResult;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Nullable
|
public @Nullable InventoryClickResult shiftClick(AbstractInventory targetInventory, @Nullable Inventory inventory, @NotNull Player player, int slot,
|
||||||
public InventoryClickResult shiftClick(@Nullable Inventory inventory, @NotNull Player player, int slot,
|
@NotNull ItemStack clicked, @NotNull ItemStack cursor) {
|
||||||
@NotNull ItemStack clicked, @NotNull ItemStack cursor, @NotNull InventoryClickLoopHandler... loopHandlers) {
|
|
||||||
InventoryClickResult clickResult = startCondition(inventory, player, slot, ClickType.START_SHIFT_CLICK, clicked, cursor);
|
InventoryClickResult clickResult = startCondition(inventory, player, slot, ClickType.START_SHIFT_CLICK, clicked, cursor);
|
||||||
|
|
||||||
if (clickResult.isCancel()) {
|
if (clickResult.isCancel()) {
|
||||||
@ -179,72 +184,36 @@ public class InventoryClickProcessor {
|
|||||||
if (clicked.isAir())
|
if (clicked.isAir())
|
||||||
return null;
|
return null;
|
||||||
|
|
||||||
final StackingRule clickedRule = clicked.getStackingRule();
|
var pair = TransactionType.ADD.process(targetInventory, clicked, (index, itemStack) -> {
|
||||||
|
InventoryClickResult result = startCondition(inventory, player, index, ClickType.SHIFT_CLICK, itemStack, cursor);
|
||||||
|
return !result.isCancel();
|
||||||
|
});
|
||||||
|
|
||||||
boolean filled = false;
|
ItemStack itemResult = TransactionOption.ALL.fill(targetInventory, pair.left(), pair.right());
|
||||||
ItemStack resultClicked = clicked;
|
clickResult.setClicked(itemResult);
|
||||||
|
return clickResult;
|
||||||
for (InventoryClickLoopHandler loopHandler : loopHandlers) {
|
|
||||||
final Int2IntFunction indexModifier = loopHandler.getIndexModifier();
|
|
||||||
final Int2ObjectFunction<ItemStack> itemGetter = loopHandler.getItemGetter();
|
|
||||||
final BiConsumer<Integer, ItemStack> itemSetter = loopHandler.getItemSetter();
|
|
||||||
|
|
||||||
for (int i = loopHandler.getStart(); i < loopHandler.getEnd(); i += loopHandler.getStep()) {
|
|
||||||
final int index = indexModifier.apply(i);
|
|
||||||
if (index == slot)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
ItemStack item = itemGetter.apply(index);
|
|
||||||
final StackingRule itemRule = item.getStackingRule();
|
|
||||||
if (itemRule.canBeStacked(item, clicked)) {
|
|
||||||
|
|
||||||
clickResult = startCondition(inventory, player, index, ClickType.SHIFT_CLICK, item, cursor);
|
|
||||||
if (clickResult.isCancel())
|
|
||||||
break;
|
|
||||||
|
|
||||||
final int amount = itemRule.getAmount(item);
|
|
||||||
if (!clickedRule.canApply(clicked, amount + 1))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
final int totalAmount = clickedRule.getAmount(resultClicked) + amount;
|
|
||||||
if (!clickedRule.canApply(clicked, totalAmount)) {
|
|
||||||
item = itemRule.apply(item, itemRule.getMaxSize());
|
|
||||||
itemSetter.accept(index, item);
|
|
||||||
|
|
||||||
resultClicked = clickedRule.apply(resultClicked, totalAmount - clickedRule.getMaxSize());
|
|
||||||
filled = false;
|
|
||||||
|
|
||||||
callClickEvent(player, inventory, index, ClickType.SHIFT_CLICK, item, cursor);
|
|
||||||
continue;
|
|
||||||
} else {
|
|
||||||
resultClicked = clickedRule.apply(resultClicked, totalAmount);
|
|
||||||
itemSetter.accept(index, resultClicked);
|
|
||||||
|
|
||||||
item = itemRule.apply(item, 0);
|
|
||||||
itemSetter.accept(slot, item);
|
|
||||||
filled = true;
|
|
||||||
|
|
||||||
callClickEvent(player, inventory, index, ClickType.SHIFT_CLICK, item, cursor);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
} else if (item.isAir()) {
|
|
||||||
|
|
||||||
clickResult = startCondition(inventory, player, index, ClickType.SHIFT_CLICK, item, cursor);
|
|
||||||
if (clickResult.isCancel())
|
|
||||||
break;
|
|
||||||
|
|
||||||
// Switch
|
|
||||||
itemSetter.accept(index, resultClicked);
|
|
||||||
itemSetter.accept(slot, ItemStack.AIR);
|
|
||||||
filled = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (!filled) {
|
|
||||||
itemSetter.accept(slot, resultClicked);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public @Nullable InventoryClickResult shiftClick(@NotNull AbstractInventory targetInventory, int start, int end, int step, @NotNull Player player, int slot,
|
||||||
|
@NotNull ItemStack clicked, @NotNull ItemStack cursor) {
|
||||||
|
InventoryClickResult clickResult = startCondition(null, player, slot, ClickType.START_SHIFT_CLICK, clicked, cursor);
|
||||||
|
|
||||||
|
if (clickResult.isCancel()) {
|
||||||
|
return clickResult;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (clicked.isAir())
|
||||||
|
return null;
|
||||||
|
|
||||||
|
var pair = TransactionType.ADD.process(targetInventory, clicked, (index, itemStack) -> {
|
||||||
|
if (index == slot) // Prevent item lose/duplication
|
||||||
|
return false;
|
||||||
|
InventoryClickResult result = startCondition(null, player, index, ClickType.SHIFT_CLICK, itemStack, cursor);
|
||||||
|
return !result.isCancel();
|
||||||
|
}, start, end, step);
|
||||||
|
|
||||||
|
ItemStack itemResult = TransactionOption.ALL.fill(targetInventory, pair.left(), pair.right());
|
||||||
|
clickResult.setClicked(itemResult);
|
||||||
return clickResult;
|
return clickResult;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -369,8 +338,8 @@ public class InventoryClickProcessor {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Nullable
|
@Nullable
|
||||||
public InventoryClickResult doubleClick(@Nullable Inventory inventory, @NotNull Player player, int slot,
|
public InventoryClickResult doubleClick(@NotNull AbstractInventory clickedInventory, @Nullable Inventory inventory, @NotNull Player player, int slot,
|
||||||
@NotNull ItemStack cursor, @NotNull InventoryClickLoopHandler... loopHandlers) {
|
@NotNull final ItemStack cursor) {
|
||||||
InventoryClickResult clickResult = startCondition(inventory, player, slot, ClickType.START_DOUBLE_CLICK, ItemStack.AIR, cursor);
|
InventoryClickResult clickResult = startCondition(inventory, player, slot, ClickType.START_DOUBLE_CLICK, ItemStack.AIR, cursor);
|
||||||
|
|
||||||
if (clickResult.isCancel()) {
|
if (clickResult.isCancel()) {
|
||||||
@ -381,49 +350,46 @@ public class InventoryClickProcessor {
|
|||||||
return null;
|
return null;
|
||||||
|
|
||||||
final StackingRule cursorRule = cursor.getStackingRule();
|
final StackingRule cursorRule = cursor.getStackingRule();
|
||||||
int amount = cursorRule.getAmount(cursor);
|
final int amount = cursorRule.getAmount(cursor);
|
||||||
|
final int remainingAmount = cursorRule.getMaxSize() - amount;
|
||||||
|
|
||||||
if (!cursorRule.canApply(cursor, amount + 1))
|
ItemStack remain = cursorRule.apply(cursor, remainingAmount);
|
||||||
return null;
|
|
||||||
|
|
||||||
for (InventoryClickLoopHandler loopHandler : loopHandlers) {
|
BiFunction<AbstractInventory, ItemStack, ItemStack> func = (inv, rest) -> {
|
||||||
final Int2IntFunction indexModifier = loopHandler.getIndexModifier();
|
var pair = TransactionType.TAKE.process(inv, rest, (index, itemStack) -> {
|
||||||
final Int2ObjectFunction<ItemStack> itemGetter = loopHandler.getItemGetter();
|
if (index == slot) // Prevent item lose/duplication
|
||||||
final BiConsumer<Integer, ItemStack> itemSetter = loopHandler.getItemSetter();
|
return false;
|
||||||
|
InventoryClickResult result = startCondition(inventory, player, index, ClickType.DOUBLE_CLICK, itemStack, cursor);
|
||||||
|
return !result.isCancel();
|
||||||
|
});
|
||||||
|
var itemResult = pair.left();
|
||||||
|
var map = pair.right();
|
||||||
|
return TransactionOption.ALL.fill(inv, itemResult, map);
|
||||||
|
};
|
||||||
|
|
||||||
for (int i = loopHandler.getStart(); i < loopHandler.getEnd(); i += loopHandler.getStep()) {
|
var playerInventory = player.getInventory();
|
||||||
final int index = indexModifier.apply(i);
|
|
||||||
if (index == slot)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
ItemStack item = itemGetter.apply(index);
|
if (Objects.equals(clickedInventory, inventory)) {
|
||||||
final StackingRule itemRule = item.getStackingRule();
|
// Clicked inside inventory
|
||||||
if (!cursorRule.canApply(cursor, amount + 1))
|
remain = func.apply(inventory, remain);
|
||||||
break;
|
if (!remain.isAir()) {
|
||||||
if (cursorRule.canBeStacked(cursor, item)) {
|
remain = func.apply(playerInventory, remain);
|
||||||
clickResult = startCondition(inventory, player, index, ClickType.DOUBLE_CLICK, item, cursor);
|
}
|
||||||
if (clickResult.isCancel())
|
} else if (inventory != null && clickedInventory == playerInventory) {
|
||||||
break;
|
// Clicked inside player inventory, but with another inventory open
|
||||||
|
remain = func.apply(playerInventory, remain);
|
||||||
final int totalAmount = amount + cursorRule.getAmount(item);
|
if (!remain.isAir()) {
|
||||||
if (!cursorRule.canApply(cursor, totalAmount)) {
|
remain = func.apply(inventory, remain);
|
||||||
cursor = cursorRule.apply(cursor, cursorRule.getMaxSize());
|
}
|
||||||
|
|
||||||
item = itemRule.apply(item, totalAmount - itemRule.getMaxSize());
|
|
||||||
} else {
|
} else {
|
||||||
cursor = cursorRule.apply(cursor, totalAmount);
|
// Clicked inside player inventory
|
||||||
item = itemRule.apply(item, 0);
|
remain = func.apply(playerInventory, remain);
|
||||||
}
|
|
||||||
itemSetter.accept(index, item);
|
|
||||||
amount = cursorRule.getAmount(cursor);
|
|
||||||
|
|
||||||
callClickEvent(player, inventory, index, ClickType.DOUBLE_CLICK, item, cursor);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
clickResult.setCursor(cursor);
|
final int tookAmount = remainingAmount - cursorRule.getAmount(remain);
|
||||||
|
|
||||||
|
ItemStack resultCursor = cursorRule.apply(cursor, amount + tookAmount);
|
||||||
|
clickResult.setCursor(resultCursor);
|
||||||
return clickResult;
|
return clickResult;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -71,10 +71,7 @@ public class ClientChannel extends SimpleChannelInboundHandler<InboundPacket> {
|
|||||||
|
|
||||||
// Release tick buffer
|
// Release tick buffer
|
||||||
if (playerConnection instanceof NettyPlayerConnection) {
|
if (playerConnection instanceof NettyPlayerConnection) {
|
||||||
final ByteBuf tickBuffer = ((NettyPlayerConnection) playerConnection).getTickBuffer();
|
((NettyPlayerConnection) playerConnection).releaseTickBuffer();
|
||||||
synchronized (tickBuffer) {
|
|
||||||
tickBuffer.release();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -62,7 +62,8 @@ public class NettyPlayerConnection extends PlayerConnection {
|
|||||||
private PlayerSkin bungeeSkin;
|
private PlayerSkin bungeeSkin;
|
||||||
|
|
||||||
private final static int INITIAL_BUFFER_SIZE = 65_535; // 2^16-1
|
private final static int INITIAL_BUFFER_SIZE = 65_535; // 2^16-1
|
||||||
private final ByteBuf tickBuffer = BufUtils.getBuffer(true);
|
private final Object tickBufferLock = new Object();
|
||||||
|
private volatile ByteBuf tickBuffer = BufUtils.getBuffer(true);
|
||||||
|
|
||||||
public NettyPlayerConnection(@NotNull SocketChannel channel) {
|
public NettyPlayerConnection(@NotNull SocketChannel channel) {
|
||||||
super();
|
super();
|
||||||
@ -119,7 +120,7 @@ public class NettyPlayerConnection extends PlayerConnection {
|
|||||||
if (getPlayer() != null) {
|
if (getPlayer() != null) {
|
||||||
// Flush happen during #update()
|
// Flush happen during #update()
|
||||||
if (serverPacket instanceof CacheablePacket && MinecraftServer.hasPacketCaching()) {
|
if (serverPacket instanceof CacheablePacket && MinecraftServer.hasPacketCaching()) {
|
||||||
synchronized (tickBuffer) {
|
synchronized (tickBufferLock) {
|
||||||
if (tickBuffer.refCnt() == 0)
|
if (tickBuffer.refCnt() == 0)
|
||||||
return;
|
return;
|
||||||
CacheablePacket.writeCache(tickBuffer, serverPacket);
|
CacheablePacket.writeCache(tickBuffer, serverPacket);
|
||||||
@ -141,7 +142,7 @@ public class NettyPlayerConnection extends PlayerConnection {
|
|||||||
public void write(@NotNull Object message, boolean skipTranslating) {
|
public void write(@NotNull Object message, boolean skipTranslating) {
|
||||||
if (message instanceof FramedPacket) {
|
if (message instanceof FramedPacket) {
|
||||||
final FramedPacket framedPacket = (FramedPacket) message;
|
final FramedPacket framedPacket = (FramedPacket) message;
|
||||||
synchronized (tickBuffer) {
|
synchronized (tickBufferLock) {
|
||||||
if (tickBuffer.refCnt() == 0)
|
if (tickBuffer.refCnt() == 0)
|
||||||
return;
|
return;
|
||||||
final ByteBuf body = framedPacket.getBody();
|
final ByteBuf body = framedPacket.getBody();
|
||||||
@ -155,14 +156,14 @@ public class NettyPlayerConnection extends PlayerConnection {
|
|||||||
serverPacket = ((ComponentHoldingServerPacket) serverPacket).copyWithOperator(component -> AdventureSerializer.translate(component, getPlayer()));
|
serverPacket = ((ComponentHoldingServerPacket) serverPacket).copyWithOperator(component -> AdventureSerializer.translate(component, getPlayer()));
|
||||||
}
|
}
|
||||||
|
|
||||||
synchronized (tickBuffer) {
|
synchronized (tickBufferLock) {
|
||||||
if (tickBuffer.refCnt() == 0)
|
if (tickBuffer.refCnt() == 0)
|
||||||
return;
|
return;
|
||||||
PacketUtils.writeFramedPacket(tickBuffer, serverPacket);
|
PacketUtils.writeFramedPacket(tickBuffer, serverPacket);
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
} else if (message instanceof ByteBuf) {
|
} else if (message instanceof ByteBuf) {
|
||||||
synchronized (tickBuffer) {
|
synchronized (tickBufferLock) {
|
||||||
if (tickBuffer.refCnt() == 0)
|
if (tickBuffer.refCnt() == 0)
|
||||||
return;
|
return;
|
||||||
tickBuffer.writeBytes((ByteBuf) message);
|
tickBuffer.writeBytes((ByteBuf) message);
|
||||||
@ -193,11 +194,11 @@ public class NettyPlayerConnection extends PlayerConnection {
|
|||||||
|
|
||||||
// Retrieve safe copy
|
// Retrieve safe copy
|
||||||
final ByteBuf copy;
|
final ByteBuf copy;
|
||||||
synchronized (tickBuffer) {
|
synchronized (tickBufferLock) {
|
||||||
if (tickBuffer.refCnt() == 0)
|
if (tickBuffer.refCnt() == 0)
|
||||||
return;
|
return;
|
||||||
copy = tickBuffer.copy();
|
copy = tickBuffer;
|
||||||
tickBuffer.clear();
|
tickBuffer = tickBuffer.alloc().buffer(tickBuffer.writerIndex());
|
||||||
}
|
}
|
||||||
|
|
||||||
// Write copied buffer to netty
|
// Write copied buffer to netty
|
||||||
@ -379,10 +380,10 @@ public class NettyPlayerConnection extends PlayerConnection {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void releaseTickBuffer() {
|
||||||
@NotNull
|
synchronized (tickBufferLock) {
|
||||||
public ByteBuf getTickBuffer() {
|
tickBuffer.release();
|
||||||
return tickBuffer;
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public byte[] getNonce() {
|
public byte[] getNonce() {
|
||||||
|
Loading…
Reference in New Issue
Block a user