mirror of
https://github.com/pmmp/PocketMine-MP.git
synced 2025-07-06 10:01:53 +00:00
Fixed handling of fake requests during block placement and other actions
This commit is contained in:
parent
2b7510945a
commit
3235d128e5
@ -339,6 +339,17 @@ class InGamePacketHandler extends PacketHandler{
|
|||||||
}else{
|
}else{
|
||||||
$this->inventoryManager->syncMismatchedPredictedSlotChanges();
|
$this->inventoryManager->syncMismatchedPredictedSlotChanges();
|
||||||
}
|
}
|
||||||
|
if($packet->requestId !== 0){
|
||||||
|
$itemStackResponseBuilder = new ItemStackResponseBuilder($packet->requestId, $this->inventoryManager);
|
||||||
|
foreach($packet->requestChangedSlots as $requestChangedSlotsEntry){
|
||||||
|
foreach($requestChangedSlotsEntry->getChangedSlotIndexes() as $slotId){
|
||||||
|
$itemStackResponseBuilder->addSlot($requestChangedSlotsEntry->getContainerId(), $slotId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$itemStackResponse = $itemStackResponseBuilder->build($result);
|
||||||
|
$this->session->sendDataPacket(ItemStackResponsePacket::create([$itemStackResponse]));
|
||||||
|
$this->session->getLogger()->debug("Sent item stack response for fake stack request " . $packet->requestId);
|
||||||
|
}
|
||||||
return $result;
|
return $result;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -551,6 +562,7 @@ class InGamePacketHandler extends PacketHandler{
|
|||||||
$executor = new ItemStackRequestExecutor($this->player, $this->inventoryManager, $request);
|
$executor = new ItemStackRequestExecutor($this->player, $this->inventoryManager, $request);
|
||||||
$transaction = $executor->generateInventoryTransaction();
|
$transaction = $executor->generateInventoryTransaction();
|
||||||
$result = $this->executeInventoryTransaction($transaction);
|
$result = $this->executeInventoryTransaction($transaction);
|
||||||
|
$this->session->getLogger()->debug("Item stack request " . $request->getRequestId() . " result: " . ($result ? "success" : "failure"));
|
||||||
$responses[] = $executor->buildItemStackResponse($result);
|
$responses[] = $executor->buildItemStackResponse($result);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
90
src/network/mcpe/handler/ItemStackContainerIdTranslator.php
Normal file
90
src/network/mcpe/handler/ItemStackContainerIdTranslator.php
Normal file
@ -0,0 +1,90 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/*
|
||||||
|
*
|
||||||
|
* ____ _ _ __ __ _ __ __ ____
|
||||||
|
* | _ \ ___ ___| | _____| |_| \/ (_)_ __ ___ | \/ | _ \
|
||||||
|
* | |_) / _ \ / __| |/ / _ \ __| |\/| | | '_ \ / _ \_____| |\/| | |_) |
|
||||||
|
* | __/ (_) | (__| < __/ |_| | | | | | | | __/_____| | | | __/
|
||||||
|
* |_| \___/ \___|_|\_\___|\__|_| |_|_|_| |_|\___| |_| |_|_|
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Lesser General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* @author PocketMine Team
|
||||||
|
* @link http://www.pocketmine.net/
|
||||||
|
*
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace pocketmine\network\mcpe\handler;
|
||||||
|
|
||||||
|
use pocketmine\network\mcpe\protocol\types\inventory\ContainerIds;
|
||||||
|
use pocketmine\network\mcpe\protocol\types\inventory\ContainerUIIds;
|
||||||
|
use pocketmine\network\PacketHandlingException;
|
||||||
|
|
||||||
|
final class ItemStackContainerIdTranslator{
|
||||||
|
|
||||||
|
private function __construct(){
|
||||||
|
//NOOP
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function translate(int $containerInterfaceId, int $currentWindowId) : int{
|
||||||
|
return match($containerInterfaceId){
|
||||||
|
ContainerUIIds::ARMOR => ContainerIds::ARMOR,
|
||||||
|
|
||||||
|
ContainerUIIds::HOTBAR,
|
||||||
|
ContainerUIIds::INVENTORY,
|
||||||
|
ContainerUIIds::COMBINED_HOTBAR_AND_INVENTORY => ContainerIds::INVENTORY,
|
||||||
|
|
||||||
|
ContainerUIIds::OFFHAND => ContainerIds::OFFHAND,
|
||||||
|
|
||||||
|
ContainerUIIds::ANVIL_INPUT,
|
||||||
|
ContainerUIIds::ANVIL_MATERIAL,
|
||||||
|
ContainerUIIds::BEACON_PAYMENT,
|
||||||
|
ContainerUIIds::CARTOGRAPHY_ADDITIONAL,
|
||||||
|
ContainerUIIds::CARTOGRAPHY_INPUT,
|
||||||
|
ContainerUIIds::COMPOUND_CREATOR_INPUT,
|
||||||
|
ContainerUIIds::CRAFTING_INPUT,
|
||||||
|
ContainerUIIds::CREATED_OUTPUT,
|
||||||
|
ContainerUIIds::CURSOR,
|
||||||
|
ContainerUIIds::ENCHANTING_INPUT,
|
||||||
|
ContainerUIIds::ENCHANTING_MATERIAL,
|
||||||
|
ContainerUIIds::GRINDSTONE_ADDITIONAL,
|
||||||
|
ContainerUIIds::GRINDSTONE_INPUT,
|
||||||
|
ContainerUIIds::LAB_TABLE_INPUT,
|
||||||
|
ContainerUIIds::LOOM_DYE,
|
||||||
|
ContainerUIIds::LOOM_INPUT,
|
||||||
|
ContainerUIIds::LOOM_MATERIAL,
|
||||||
|
ContainerUIIds::MATERIAL_REDUCER_INPUT,
|
||||||
|
ContainerUIIds::MATERIAL_REDUCER_OUTPUT,
|
||||||
|
ContainerUIIds::SMITHING_TABLE_INPUT,
|
||||||
|
ContainerUIIds::SMITHING_TABLE_MATERIAL,
|
||||||
|
ContainerUIIds::STONECUTTER_INPUT,
|
||||||
|
ContainerUIIds::TRADE2_INGREDIENT1,
|
||||||
|
ContainerUIIds::TRADE2_INGREDIENT2,
|
||||||
|
ContainerUIIds::TRADE_INGREDIENT1,
|
||||||
|
ContainerUIIds::TRADE_INGREDIENT2 => ContainerIds::UI,
|
||||||
|
|
||||||
|
ContainerUIIds::BARREL,
|
||||||
|
ContainerUIIds::BLAST_FURNACE_INGREDIENT,
|
||||||
|
ContainerUIIds::BREWING_STAND_FUEL,
|
||||||
|
ContainerUIIds::BREWING_STAND_INPUT,
|
||||||
|
ContainerUIIds::BREWING_STAND_RESULT,
|
||||||
|
ContainerUIIds::FURNACE_FUEL,
|
||||||
|
ContainerUIIds::FURNACE_INGREDIENT,
|
||||||
|
ContainerUIIds::FURNACE_RESULT,
|
||||||
|
ContainerUIIds::LEVEL_ENTITY, //chest
|
||||||
|
ContainerUIIds::SHULKER_BOX,
|
||||||
|
ContainerUIIds::SMOKER_INGREDIENT => $currentWindowId,
|
||||||
|
|
||||||
|
//all preview slots are ignored, since the client shouldn't be modifying those directly
|
||||||
|
|
||||||
|
default => throw new PacketHandlingException("Unexpected container UI ID $containerInterfaceId")
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
@ -23,8 +23,6 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
namespace pocketmine\network\mcpe\handler;
|
namespace pocketmine\network\mcpe\handler;
|
||||||
|
|
||||||
use pocketmine\block\inventory\CraftingTableInventory;
|
|
||||||
use pocketmine\inventory\Inventory;
|
|
||||||
use pocketmine\inventory\transaction\action\DestroyItemAction;
|
use pocketmine\inventory\transaction\action\DestroyItemAction;
|
||||||
use pocketmine\inventory\transaction\action\DropItemAction;
|
use pocketmine\inventory\transaction\action\DropItemAction;
|
||||||
use pocketmine\inventory\transaction\CraftingTransaction;
|
use pocketmine\inventory\transaction\CraftingTransaction;
|
||||||
@ -34,8 +32,6 @@ use pocketmine\inventory\transaction\TransactionBuilderInventory;
|
|||||||
use pocketmine\item\Item;
|
use pocketmine\item\Item;
|
||||||
use pocketmine\item\VanillaItems;
|
use pocketmine\item\VanillaItems;
|
||||||
use pocketmine\network\mcpe\InventoryManager;
|
use pocketmine\network\mcpe\InventoryManager;
|
||||||
use pocketmine\network\mcpe\protocol\types\inventory\ContainerIds;
|
|
||||||
use pocketmine\network\mcpe\protocol\types\inventory\ContainerUIIds;
|
|
||||||
use pocketmine\network\mcpe\protocol\types\inventory\stackrequest\CraftingConsumeInputStackRequestAction;
|
use pocketmine\network\mcpe\protocol\types\inventory\stackrequest\CraftingConsumeInputStackRequestAction;
|
||||||
use pocketmine\network\mcpe\protocol\types\inventory\stackrequest\CraftingMarkSecondaryResultStackRequestAction;
|
use pocketmine\network\mcpe\protocol\types\inventory\stackrequest\CraftingMarkSecondaryResultStackRequestAction;
|
||||||
use pocketmine\network\mcpe\protocol\types\inventory\stackrequest\CraftRecipeAutoStackRequestAction;
|
use pocketmine\network\mcpe\protocol\types\inventory\stackrequest\CraftRecipeAutoStackRequestAction;
|
||||||
@ -50,12 +46,9 @@ use pocketmine\network\mcpe\protocol\types\inventory\stackrequest\PlaceStackRequ
|
|||||||
use pocketmine\network\mcpe\protocol\types\inventory\stackrequest\SwapStackRequestAction;
|
use pocketmine\network\mcpe\protocol\types\inventory\stackrequest\SwapStackRequestAction;
|
||||||
use pocketmine\network\mcpe\protocol\types\inventory\stackrequest\TakeStackRequestAction;
|
use pocketmine\network\mcpe\protocol\types\inventory\stackrequest\TakeStackRequestAction;
|
||||||
use pocketmine\network\mcpe\protocol\types\inventory\stackresponse\ItemStackResponse;
|
use pocketmine\network\mcpe\protocol\types\inventory\stackresponse\ItemStackResponse;
|
||||||
use pocketmine\network\mcpe\protocol\types\inventory\stackresponse\ItemStackResponseContainerInfo;
|
|
||||||
use pocketmine\network\mcpe\protocol\types\inventory\stackresponse\ItemStackResponseSlotInfo;
|
|
||||||
use pocketmine\network\PacketHandlingException;
|
use pocketmine\network\PacketHandlingException;
|
||||||
use pocketmine\player\Player;
|
use pocketmine\player\Player;
|
||||||
use function get_class;
|
use function get_class;
|
||||||
use function var_dump;
|
|
||||||
|
|
||||||
final class ItemStackRequestExecutor{
|
final class ItemStackRequestExecutor{
|
||||||
private TransactionBuilder $builder;
|
private TransactionBuilder $builder;
|
||||||
@ -73,83 +66,19 @@ final class ItemStackRequestExecutor{
|
|||||||
$this->builder = new TransactionBuilder();
|
$this->builder = new TransactionBuilder();
|
||||||
}
|
}
|
||||||
|
|
||||||
private function translateContainerId(int $containerInterfaceId) : int{
|
|
||||||
return match($containerInterfaceId){
|
|
||||||
ContainerUIIds::ARMOR => ContainerIds::ARMOR,
|
|
||||||
|
|
||||||
ContainerUIIds::HOTBAR,
|
|
||||||
ContainerUIIds::INVENTORY,
|
|
||||||
ContainerUIIds::COMBINED_HOTBAR_AND_INVENTORY => ContainerIds::INVENTORY,
|
|
||||||
|
|
||||||
ContainerUIIds::OFFHAND => ContainerIds::OFFHAND,
|
|
||||||
|
|
||||||
ContainerUIIds::ANVIL_INPUT,
|
|
||||||
ContainerUIIds::ANVIL_MATERIAL,
|
|
||||||
ContainerUIIds::BEACON_PAYMENT,
|
|
||||||
ContainerUIIds::CARTOGRAPHY_ADDITIONAL,
|
|
||||||
ContainerUIIds::CARTOGRAPHY_INPUT,
|
|
||||||
ContainerUIIds::COMPOUND_CREATOR_INPUT,
|
|
||||||
ContainerUIIds::CRAFTING_INPUT,
|
|
||||||
ContainerUIIds::CREATED_OUTPUT,
|
|
||||||
ContainerUIIds::CURSOR,
|
|
||||||
ContainerUIIds::ENCHANTING_INPUT,
|
|
||||||
ContainerUIIds::ENCHANTING_MATERIAL,
|
|
||||||
ContainerUIIds::GRINDSTONE_ADDITIONAL,
|
|
||||||
ContainerUIIds::GRINDSTONE_INPUT,
|
|
||||||
ContainerUIIds::LAB_TABLE_INPUT,
|
|
||||||
ContainerUIIds::LOOM_DYE,
|
|
||||||
ContainerUIIds::LOOM_INPUT,
|
|
||||||
ContainerUIIds::LOOM_MATERIAL,
|
|
||||||
ContainerUIIds::MATERIAL_REDUCER_INPUT,
|
|
||||||
ContainerUIIds::MATERIAL_REDUCER_OUTPUT,
|
|
||||||
ContainerUIIds::SMITHING_TABLE_INPUT,
|
|
||||||
ContainerUIIds::SMITHING_TABLE_MATERIAL,
|
|
||||||
ContainerUIIds::STONECUTTER_INPUT,
|
|
||||||
ContainerUIIds::TRADE2_INGREDIENT1,
|
|
||||||
ContainerUIIds::TRADE2_INGREDIENT2,
|
|
||||||
ContainerUIIds::TRADE_INGREDIENT1,
|
|
||||||
ContainerUIIds::TRADE_INGREDIENT2 => ContainerIds::UI,
|
|
||||||
|
|
||||||
ContainerUIIds::BARREL,
|
|
||||||
ContainerUIIds::BLAST_FURNACE_INGREDIENT,
|
|
||||||
ContainerUIIds::BREWING_STAND_FUEL,
|
|
||||||
ContainerUIIds::BREWING_STAND_INPUT,
|
|
||||||
ContainerUIIds::BREWING_STAND_RESULT,
|
|
||||||
ContainerUIIds::FURNACE_FUEL,
|
|
||||||
ContainerUIIds::FURNACE_INGREDIENT,
|
|
||||||
ContainerUIIds::FURNACE_RESULT,
|
|
||||||
ContainerUIIds::LEVEL_ENTITY, //chest
|
|
||||||
ContainerUIIds::SHULKER_BOX,
|
|
||||||
ContainerUIIds::SMOKER_INGREDIENT => $this->inventoryManager->getCurrentWindowId(),
|
|
||||||
|
|
||||||
//all preview slots are ignored, since the client shouldn't be modifying those directly
|
|
||||||
|
|
||||||
default => throw new PacketHandlingException("Unexpected container UI ID $containerInterfaceId")
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @phpstan-return array{Inventory, int}
|
|
||||||
*/
|
|
||||||
private function getInventoryAndSlot(ItemStackRequestSlotInfo $info) : array{
|
|
||||||
$windowId = $this->translateContainerId($info->getContainerId());
|
|
||||||
$info = $this->inventoryManager->locateWindowAndSlot($windowId, $info->getSlotId());
|
|
||||||
if($info === null){
|
|
||||||
throw new PacketHandlingException("Stack request action cannot target an inventory that is not open");
|
|
||||||
}
|
|
||||||
[$inventory, $slot] = $info;
|
|
||||||
if(!$inventory->slotExists($slot)){
|
|
||||||
throw new PacketHandlingException("Stack request action cannot target an inventory slot that does not exist");
|
|
||||||
}
|
|
||||||
|
|
||||||
return [$inventory, $slot];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @phpstan-return array{TransactionBuilderInventory, int}
|
* @phpstan-return array{TransactionBuilderInventory, int}
|
||||||
*/
|
*/
|
||||||
private function getBuilderInventoryAndSlot(ItemStackRequestSlotInfo $info) : array{
|
private function getBuilderInventoryAndSlot(ItemStackRequestSlotInfo $info) : array{
|
||||||
[$inventory, $slot] = $this->getInventoryAndSlot($info);
|
$windowId = ItemStackContainerIdTranslator::translate($info->getContainerId(), $this->inventoryManager->getCurrentWindowId());
|
||||||
|
$windowAndSlot = $this->inventoryManager->locateWindowAndSlot($windowId, $info->getSlotId());
|
||||||
|
if($windowAndSlot === null){
|
||||||
|
throw new PacketHandlingException("Stack request action cannot target an inventory that is not open");
|
||||||
|
}
|
||||||
|
[$inventory, $slot] = $windowAndSlot;
|
||||||
|
if(!$inventory->slotExists($slot)){
|
||||||
|
throw new PacketHandlingException("Stack request action cannot target an inventory slot that does not exist");
|
||||||
|
}
|
||||||
|
|
||||||
if(
|
if(
|
||||||
$info->getStackId() !== $this->request->getRequestId() && //using TransactionBuilderInventory enables this to work
|
$info->getStackId() !== $this->request->getRequestId() && //using TransactionBuilderInventory enables this to work
|
||||||
@ -261,28 +190,11 @@ final class ItemStackRequestExecutor{
|
|||||||
}
|
}
|
||||||
|
|
||||||
public function buildItemStackResponse(bool $success) : ItemStackResponse{
|
public function buildItemStackResponse(bool $success) : ItemStackResponse{
|
||||||
$responseInfosByContainer = [];
|
$builder = new ItemStackResponseBuilder($this->request->getRequestId(), $this->inventoryManager);
|
||||||
foreach($this->requestSlotInfos as $requestInfo){
|
foreach($this->requestSlotInfos as $requestInfo){
|
||||||
[$inventory, $slot] = $this->getInventoryAndSlot($requestInfo);
|
$builder->addSlot($requestInfo->getContainerId(), $requestInfo->getSlotId());
|
||||||
|
|
||||||
$item = $inventory->getItem($slot);
|
|
||||||
$info = $this->inventoryManager->trackItemStack($inventory, $slot, $item, $this->request->getRequestId());
|
|
||||||
|
|
||||||
$responseInfosByContainer[$requestInfo->getContainerId()][] = new ItemStackResponseSlotInfo(
|
|
||||||
$requestInfo->getSlotId(),
|
|
||||||
$requestInfo->getSlotId(),
|
|
||||||
$info->getItemStack()->getCount(),
|
|
||||||
$info->getStackId(),
|
|
||||||
$item->hasCustomName() ? $item->getCustomName() : "",
|
|
||||||
0
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
$responseContainerInfos = [];
|
return $builder->build($success);
|
||||||
foreach($responseInfosByContainer as $containerId => $responseInfos){
|
|
||||||
$responseContainerInfos[] = new ItemStackResponseContainerInfo($containerId, $responseInfos);
|
|
||||||
}
|
|
||||||
|
|
||||||
return new ItemStackResponse($success ? ItemStackResponse::RESULT_OK : ItemStackResponse::RESULT_ERROR, $this->request->getRequestId(), $responseContainerInfos);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
94
src/network/mcpe/handler/ItemStackResponseBuilder.php
Normal file
94
src/network/mcpe/handler/ItemStackResponseBuilder.php
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/*
|
||||||
|
*
|
||||||
|
* ____ _ _ __ __ _ __ __ ____
|
||||||
|
* | _ \ ___ ___| | _____| |_| \/ (_)_ __ ___ | \/ | _ \
|
||||||
|
* | |_) / _ \ / __| |/ / _ \ __| |\/| | | '_ \ / _ \_____| |\/| | |_) |
|
||||||
|
* | __/ (_) | (__| < __/ |_| | | | | | | | __/_____| | | | __/
|
||||||
|
* |_| \___/ \___|_|\_\___|\__|_| |_|_|_| |_|\___| |_| |_|_|
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Lesser General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* @author PocketMine Team
|
||||||
|
* @link http://www.pocketmine.net/
|
||||||
|
*
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace pocketmine\network\mcpe\handler;
|
||||||
|
|
||||||
|
use pocketmine\inventory\Inventory;
|
||||||
|
use pocketmine\network\mcpe\InventoryManager;
|
||||||
|
use pocketmine\network\mcpe\protocol\types\inventory\stackresponse\ItemStackResponse;
|
||||||
|
use pocketmine\network\mcpe\protocol\types\inventory\stackresponse\ItemStackResponseContainerInfo;
|
||||||
|
use pocketmine\network\mcpe\protocol\types\inventory\stackresponse\ItemStackResponseSlotInfo;
|
||||||
|
use pocketmine\network\PacketHandlingException;
|
||||||
|
|
||||||
|
final class ItemStackResponseBuilder{
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var int[][]
|
||||||
|
* @phpstan-var array<int, array<int, int>>
|
||||||
|
*/
|
||||||
|
private array $changedSlots = [];
|
||||||
|
|
||||||
|
public function __construct(
|
||||||
|
private int $requestId,
|
||||||
|
private InventoryManager $inventoryManager
|
||||||
|
){}
|
||||||
|
|
||||||
|
public function addSlot(int $containerInterfaceId, int $slotId) : void{
|
||||||
|
$this->changedSlots[$containerInterfaceId][$slotId] = $slotId;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @phpstan-return array{Inventory, int}
|
||||||
|
*/
|
||||||
|
private function getInventoryAndSlot(int $containerInterfaceId, int $slotId) : array{
|
||||||
|
$windowId = ItemStackContainerIdTranslator::translate($containerInterfaceId, $this->inventoryManager->getCurrentWindowId());
|
||||||
|
$windowAndSlot = $this->inventoryManager->locateWindowAndSlot($windowId, $slotId);
|
||||||
|
if($windowAndSlot === null){
|
||||||
|
throw new PacketHandlingException("Stack request action cannot target an inventory that is not open");
|
||||||
|
}
|
||||||
|
[$inventory, $slot] = $windowAndSlot;
|
||||||
|
if(!$inventory->slotExists($slot)){
|
||||||
|
throw new PacketHandlingException("Stack request action cannot target an inventory slot that does not exist");
|
||||||
|
}
|
||||||
|
|
||||||
|
return [$inventory, $slot];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function build(bool $success) : ItemStackResponse{
|
||||||
|
$responseInfosByContainer = [];
|
||||||
|
foreach($this->changedSlots as $containerInterfaceId => $slotIds){
|
||||||
|
foreach($slotIds as $slotId){
|
||||||
|
[$inventory, $slot] = $this->getInventoryAndSlot($containerInterfaceId, $slotId);
|
||||||
|
|
||||||
|
$item = $inventory->getItem($slot);
|
||||||
|
$info = $this->inventoryManager->trackItemStack($inventory, $slot, $item, $this->requestId);
|
||||||
|
|
||||||
|
$responseInfosByContainer[$containerInterfaceId][] = new ItemStackResponseSlotInfo(
|
||||||
|
$slotId,
|
||||||
|
$slotId,
|
||||||
|
$info->getItemStack()->getCount(),
|
||||||
|
$info->getStackId(),
|
||||||
|
$item->hasCustomName() ? $item->getCustomName() : "",
|
||||||
|
0
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$responseContainerInfos = [];
|
||||||
|
foreach($responseInfosByContainer as $containerInterfaceId => $responseInfos){
|
||||||
|
$responseContainerInfos[] = new ItemStackResponseContainerInfo($containerInterfaceId, $responseInfos);
|
||||||
|
}
|
||||||
|
|
||||||
|
return new ItemStackResponse($success ? ItemStackResponse::RESULT_OK : ItemStackResponse::RESULT_ERROR, $this->requestId, $responseContainerInfos);
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user