4jcraft/Minecraft.World/Containers/MerchantContainer.cpp
Tropical f25cd66f4d TU19: merge Minecraft.World/Containers
keeping virtual destructors where possible
2026-03-21 15:18:52 -05:00

150 lines
4.7 KiB
C++

#include "../Platform/stdafx.h"
#include "../Headers/net.minecraft.world.item.trading.h"
#include "MerchantMenu.h"
#include "MerchantContainer.h"
MerchantContainer::MerchantContainer(std::shared_ptr<Player> player,
std::shared_ptr<Merchant> villager) {
this->player = player;
merchant = villager;
items = ItemInstanceArray(3);
items[0] = nullptr;
items[1] = nullptr;
items[2] = nullptr;
activeRecipe = NULL;
selectionHint = 0;
}
MerchantContainer::~MerchantContainer() { delete[] items.data; }
unsigned int MerchantContainer::getContainerSize() { return items.length; }
std::shared_ptr<ItemInstance> MerchantContainer::getItem(unsigned int slot) {
return items[slot];
}
std::shared_ptr<ItemInstance> MerchantContainer::removeItem(unsigned int slot,
int count) {
if (items[slot] != NULL) {
if (slot == MerchantMenu::RESULT_SLOT) {
std::shared_ptr<ItemInstance> item = items[slot];
items[slot] = nullptr;
return item;
}
if (items[slot]->count <= count) {
std::shared_ptr<ItemInstance> item = items[slot];
items[slot] = nullptr;
if (isPaymentSlot(slot)) {
updateSellItem();
}
return item;
} else {
std::shared_ptr<ItemInstance> i = items[slot]->remove(count);
if (items[slot]->count == 0) items[slot] = nullptr;
if (isPaymentSlot(slot)) {
updateSellItem();
}
return i;
}
}
return nullptr;
}
bool MerchantContainer::isPaymentSlot(int slot) {
return slot == MerchantMenu::PAYMENT1_SLOT ||
slot == MerchantMenu::PAYMENT2_SLOT;
}
std::shared_ptr<ItemInstance> MerchantContainer::removeItemNoUpdate(int slot) {
if (items[slot] != NULL) {
std::shared_ptr<ItemInstance> item = items[slot];
items[slot] = nullptr;
return item;
}
return nullptr;
}
void MerchantContainer::setItem(unsigned int slot,
std::shared_ptr<ItemInstance> item) {
items[slot] = item;
if (item != NULL && item->count > getMaxStackSize())
item->count = getMaxStackSize();
if (isPaymentSlot(slot)) {
updateSellItem();
}
}
std::wstring MerchantContainer::getName() { return merchant->getDisplayName(); }
std::wstring MerchantContainer::getCustomName() { return L""; }
bool MerchantContainer::hasCustomName() { return false; }
int MerchantContainer::getMaxStackSize() {
return Container::LARGE_MAX_STACK_SIZE;
}
bool MerchantContainer::stillValid(std::shared_ptr<Player> player) {
return merchant->getTradingPlayer() == player;
}
void MerchantContainer::startOpen() {}
void MerchantContainer::stopOpen() {}
bool MerchantContainer::canPlaceItem(int slot,
std::shared_ptr<ItemInstance> item) {
return true;
}
void MerchantContainer::setChanged() { updateSellItem(); }
void MerchantContainer::updateSellItem() {
activeRecipe = NULL;
std::shared_ptr<ItemInstance> buyItem1 = items[MerchantMenu::PAYMENT1_SLOT];
std::shared_ptr<ItemInstance> buyItem2 = items[MerchantMenu::PAYMENT2_SLOT];
if (buyItem1 == NULL) {
buyItem1 = buyItem2;
buyItem2 = nullptr;
}
if (buyItem1 == NULL) {
setItem(MerchantMenu::RESULT_SLOT, nullptr);
} else {
MerchantRecipeList* offers = merchant->getOffers(player);
if (offers != NULL) {
MerchantRecipe* recipeFor =
offers->getRecipeFor(buyItem1, buyItem2, selectionHint);
if (recipeFor != NULL && !recipeFor->isDeprecated()) {
activeRecipe = recipeFor;
setItem(MerchantMenu::RESULT_SLOT,
recipeFor->getSellItem()->copy());
} else if (buyItem2 != NULL) {
// try to switch
recipeFor =
offers->getRecipeFor(buyItem2, buyItem1, selectionHint);
if (recipeFor != NULL && !recipeFor->isDeprecated()) {
activeRecipe = recipeFor;
setItem(MerchantMenu::RESULT_SLOT,
recipeFor->getSellItem()->copy());
} else {
setItem(MerchantMenu::RESULT_SLOT, nullptr);
}
} else {
setItem(MerchantMenu::RESULT_SLOT, nullptr);
}
}
}
merchant->notifyTradeUpdated(getItem(MerchantMenu::RESULT_SLOT));
}
MerchantRecipe* MerchantContainer::getActiveRecipe() { return activeRecipe; }
void MerchantContainer::setSelectionHint(int selectionHint) {
this->selectionHint = selectionHint;
updateSellItem();
}