1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-18 13:12:50 +00:00
OpenMW/apps/openmw/mwgui/tradeitemmodel.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

215 lines
6.5 KiB
C++
Raw Normal View History

2013-05-11 18:38:27 +02:00
#include "tradeitemmodel.hpp"
#include <components/misc/strings/algorithm.hpp>
#include <components/settings/settings.hpp>
2014-02-23 20:11:05 +01:00
2013-05-11 18:38:27 +02:00
#include "../mwworld/class.hpp"
#include "../mwworld/containerstore.hpp"
#include "../mwworld/inventorystore.hpp"
namespace MWGui
{
2022-08-31 18:49:50 +02:00
TradeItemModel::TradeItemModel(std::unique_ptr<ItemModel> sourceModel, const MWWorld::Ptr& merchant)
2013-05-11 18:38:27 +02:00
: mMerchant(merchant)
{
2022-08-31 18:49:50 +02:00
mSourceModel = std::move(sourceModel);
2013-05-11 18:38:27 +02:00
}
bool TradeItemModel::allowedToUseItems() const
{
return true;
}
2013-05-11 18:38:27 +02:00
ItemStack TradeItemModel::getItem(ModelIndex index)
{
if (index < 0)
throw std::runtime_error("Invalid index supplied");
if (mItems.size() <= static_cast<size_t>(index))
throw std::runtime_error("Item index out of range");
return mItems[index];
}
size_t TradeItemModel::getItemCount()
{
return mItems.size();
}
void TradeItemModel::borrowImpl(const ItemStack& item, std::vector<ItemStack>& out)
{
bool found = false;
for (ItemStack& itemStack : out)
2013-05-11 18:38:27 +02:00
{
if (itemStack.mBase == item.mBase)
2013-05-11 18:38:27 +02:00
{
itemStack.mCount += item.mCount;
2013-05-11 18:38:27 +02:00
found = true;
break;
}
}
if (!found)
out.push_back(item);
}
void TradeItemModel::unborrowImpl(const ItemStack& item, size_t count, std::vector<ItemStack>& out)
{
std::vector<ItemStack>::iterator it = out.begin();
bool found = false;
for (; it != out.end(); ++it)
{
if (it->mBase == item.mBase)
2013-05-11 18:38:27 +02:00
{
if (it->mCount < count)
throw std::runtime_error("Not enough borrowed items to return");
it->mCount -= count;
if (it->mCount == 0)
out.erase(it);
found = true;
break;
}
}
if (!found)
throw std::runtime_error("Can't find borrowed item to return");
}
void TradeItemModel::borrowItemFromUs(ModelIndex itemIndex, size_t count)
{
ItemStack item = getItem(itemIndex);
item.mCount = count;
borrowImpl(item, mBorrowedFromUs);
}
void TradeItemModel::borrowItemToUs(ModelIndex itemIndex, ItemModel* source, size_t count)
{
ItemStack item = source->getItem(itemIndex);
item.mCount = count;
borrowImpl(item, mBorrowedToUs);
}
void TradeItemModel::returnItemBorrowedToUs(ModelIndex itemIndex, size_t count)
{
ItemStack item = getItem(itemIndex);
unborrowImpl(item, count, mBorrowedToUs);
}
void TradeItemModel::returnItemBorrowedFromUs(ModelIndex itemIndex, ItemModel* source, size_t count)
{
ItemStack item = source->getItem(itemIndex);
unborrowImpl(item, count, mBorrowedFromUs);
}
void TradeItemModel::adjustEncumbrance(float& encumbrance)
{
for (ItemStack& itemStack : mBorrowedToUs)
{
MWWorld::Ptr& item = itemStack.mBase;
encumbrance += item.getClass().getWeight(item) * itemStack.mCount;
}
for (ItemStack& itemStack : mBorrowedFromUs)
{
MWWorld::Ptr& item = itemStack.mBase;
encumbrance -= item.getClass().getWeight(item) * itemStack.mCount;
}
encumbrance = std::max(0.f, encumbrance);
}
2013-05-11 18:38:27 +02:00
void TradeItemModel::abort()
{
mBorrowedFromUs.clear();
mBorrowedToUs.clear();
}
const std::vector<ItemStack> TradeItemModel::getItemsBorrowedToUs() const
2013-05-11 18:38:27 +02:00
{
return mBorrowedToUs;
}
void TradeItemModel::transferItems()
2013-05-11 18:38:27 +02:00
{
for (ItemStack& itemStack : mBorrowedToUs)
2013-05-11 18:38:27 +02:00
{
// get index in the source model
ItemModel* sourceModel = itemStack.mCreator;
2013-05-11 18:38:27 +02:00
size_t i = 0;
for (; i < sourceModel->getItemCount(); ++i)
{
if (itemStack.mBase == sourceModel->getItem(i).mBase)
2013-05-11 18:38:27 +02:00
break;
}
if (i == sourceModel->getItemCount())
throw std::runtime_error("The borrowed item disappeared");
const ItemStack& item = sourceModel->getItem(i);
static const bool prevent = Settings::Manager::getBool("prevent merchant equipping", "Game");
2013-05-11 18:38:27 +02:00
// copy the borrowed items to our model
copyItem(item, itemStack.mCount, !prevent);
2013-05-11 18:38:27 +02:00
// then remove them from the source model
sourceModel->removeItem(item, itemStack.mCount);
2013-05-11 18:38:27 +02:00
}
mBorrowedToUs.clear();
mBorrowedFromUs.clear();
}
void TradeItemModel::update()
{
mSourceModel->update();
int services = 0;
if (!mMerchant.isEmpty())
services = mMerchant.getClass().getServices(mMerchant);
2013-05-11 18:38:27 +02:00
mItems.clear();
// add regular items
for (size_t i = 0; i < mSourceModel->getItemCount(); ++i)
{
ItemStack item = mSourceModel->getItem(i);
if (!mMerchant.isEmpty())
2013-05-11 18:38:27 +02:00
{
MWWorld::Ptr base = item.mBase;
if (base.getCellRef().getRefId() == MWWorld::ContainerStore::sGoldId)
continue;
if (!base.getClass().showsInInventory(base))
return;
if (!base.getClass().canSell(base, services))
continue;
// Bound items may not be bought
if (item.mFlags & ItemStack::Flag_Bound)
continue;
// don't show equipped items
if (mMerchant.getClass().hasInventoryStore(mMerchant))
2013-05-11 18:38:27 +02:00
{
MWWorld::InventoryStore& store = mMerchant.getClass().getInventoryStore(mMerchant);
if (store.isEquipped(base))
2013-05-11 18:38:27 +02:00
continue;
}
}
// don't show items that we borrowed to someone else
for (ItemStack& itemStack : mBorrowedFromUs)
2013-05-11 18:38:27 +02:00
{
if (itemStack.mBase == item.mBase)
2013-05-11 18:38:27 +02:00
{
if (item.mCount < itemStack.mCount)
2013-05-11 18:38:27 +02:00
throw std::runtime_error("Lent more items than present");
item.mCount -= itemStack.mCount;
2013-05-11 18:38:27 +02:00
}
}
if (item.mCount > 0)
mItems.push_back(item);
}
// add items borrowed to us
for (ItemStack& itemStack : mBorrowedToUs)
2013-05-11 18:38:27 +02:00
{
itemStack.mType = ItemStack::Type_Barter;
mItems.push_back(itemStack);
2013-05-11 18:38:27 +02:00
}
}
}