2013-05-11 18:38:27 +02:00
|
|
|
#include "tradeitemmodel.hpp"
|
|
|
|
|
2014-02-23 20:11:05 +01:00
|
|
|
#include <components/misc/stringops.hpp>
|
|
|
|
|
2013-05-11 18:38:27 +02:00
|
|
|
#include "../mwworld/class.hpp"
|
|
|
|
#include "../mwworld/containerstore.hpp"
|
|
|
|
#include "../mwworld/inventorystore.hpp"
|
|
|
|
|
|
|
|
namespace MWGui
|
|
|
|
{
|
|
|
|
|
|
|
|
TradeItemModel::TradeItemModel(ItemModel *sourceModel, const MWWorld::Ptr& merchant)
|
|
|
|
: mMerchant(merchant)
|
|
|
|
{
|
|
|
|
mSourceModel = sourceModel;
|
|
|
|
}
|
|
|
|
|
|
|
|
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)
|
|
|
|
{
|
|
|
|
std::vector<ItemStack>::iterator it = out.begin();
|
|
|
|
bool found = false;
|
|
|
|
for (; it != out.end(); ++it)
|
|
|
|
{
|
|
|
|
if (it->stacks(item))
|
|
|
|
{
|
|
|
|
it->mCount += item.mCount;
|
|
|
|
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->stacks(item))
|
|
|
|
{
|
|
|
|
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::abort()
|
|
|
|
{
|
|
|
|
mBorrowedFromUs.clear();
|
|
|
|
mBorrowedToUs.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<ItemStack> TradeItemModel::getItemsBorrowedToUs()
|
|
|
|
{
|
|
|
|
return mBorrowedToUs;
|
|
|
|
}
|
|
|
|
|
|
|
|
void TradeItemModel::transferItems()
|
|
|
|
{
|
|
|
|
std::vector<ItemStack>::iterator it = mBorrowedToUs.begin();
|
|
|
|
for (; it != mBorrowedToUs.end(); ++it)
|
|
|
|
{
|
|
|
|
// get index in the source model
|
|
|
|
ItemModel* sourceModel = it->mCreator;
|
|
|
|
size_t i=0;
|
|
|
|
for (; i<sourceModel->getItemCount(); ++i)
|
|
|
|
{
|
|
|
|
if (it->stacks(sourceModel->getItem(i)))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (i == sourceModel->getItemCount())
|
|
|
|
throw std::runtime_error("The borrowed item disappeared");
|
|
|
|
|
2014-04-27 05:40:07 +02:00
|
|
|
// reset owner while copying, but only for items bought by the player
|
|
|
|
bool setNewOwner = (mMerchant.isEmpty());
|
2013-05-11 18:38:27 +02:00
|
|
|
const ItemStack& item = sourceModel->getItem(i);
|
|
|
|
// copy the borrowed items to our model
|
2014-04-27 05:40:07 +02:00
|
|
|
copyItem(item, it->mCount, setNewOwner);
|
2013-05-11 18:38:27 +02:00
|
|
|
// then remove them from the source model
|
|
|
|
sourceModel->removeItem(item, it->mCount);
|
|
|
|
}
|
|
|
|
mBorrowedToUs.clear();
|
|
|
|
mBorrowedFromUs.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void TradeItemModel::update()
|
|
|
|
{
|
|
|
|
mSourceModel->update();
|
|
|
|
|
|
|
|
int services = 0;
|
|
|
|
if (!mMerchant.isEmpty())
|
|
|
|
services = MWWorld::Class::get(mMerchant).getServices(mMerchant);
|
|
|
|
|
|
|
|
mItems.clear();
|
|
|
|
// add regular items
|
|
|
|
for (size_t i=0; i<mSourceModel->getItemCount(); ++i)
|
|
|
|
{
|
|
|
|
ItemStack item = mSourceModel->getItem(i);
|
2013-08-15 04:52:01 -07:00
|
|
|
if(!mMerchant.isEmpty())
|
2013-05-11 18:38:27 +02:00
|
|
|
{
|
2013-08-15 04:52:01 -07:00
|
|
|
MWWorld::Ptr base = item.mBase;
|
2014-01-08 23:37:46 +01:00
|
|
|
if(Misc::StringUtils::ciEqual(base.getCellRef().mRefID, MWWorld::ContainerStore::sGoldId))
|
2013-08-15 04:52:01 -07:00
|
|
|
continue;
|
|
|
|
if(!MWWorld::Class::get(base).canSell(base, services))
|
|
|
|
continue;
|
|
|
|
|
2014-01-02 01:31:06 +01:00
|
|
|
// Bound items may not be bought
|
|
|
|
if (item.mBase.getCellRef().mRefID.size() > 6
|
|
|
|
&& item.mBase.getCellRef().mRefID.substr(0,6) == "bound_")
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2013-08-15 04:52:01 -07:00
|
|
|
// don't show equipped items
|
2014-01-19 11:42:58 +01:00
|
|
|
if(mMerchant.getClass().hasInventoryStore(mMerchant))
|
2013-05-11 18:38:27 +02:00
|
|
|
{
|
2013-08-15 04:52:01 -07:00
|
|
|
bool isEquipped = false;
|
|
|
|
MWWorld::InventoryStore& store = MWWorld::Class::get(mMerchant).getInventoryStore(mMerchant);
|
|
|
|
for (int slot=0; slot<MWWorld::InventoryStore::Slots; ++slot)
|
|
|
|
{
|
|
|
|
MWWorld::ContainerStoreIterator equipped = store.getSlot(slot);
|
|
|
|
if (equipped == store.end())
|
|
|
|
continue;
|
|
|
|
if (*equipped == base)
|
|
|
|
isEquipped = true;
|
|
|
|
}
|
|
|
|
if (isEquipped)
|
2013-05-11 18:38:27 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// don't show items that we borrowed to someone else
|
|
|
|
std::vector<ItemStack>::iterator it = mBorrowedFromUs.begin();
|
|
|
|
for (; it != mBorrowedFromUs.end(); ++it)
|
|
|
|
{
|
|
|
|
if (it->stacks(item))
|
|
|
|
{
|
|
|
|
if (item.mCount < it->mCount)
|
|
|
|
throw std::runtime_error("Lent more items than present");
|
|
|
|
item.mCount -= it->mCount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (item.mCount > 0)
|
|
|
|
mItems.push_back(item);
|
|
|
|
}
|
|
|
|
|
|
|
|
// add items borrowed to us
|
|
|
|
std::vector<ItemStack>::iterator it = mBorrowedToUs.begin();
|
|
|
|
for (; it != mBorrowedToUs.end(); ++it)
|
|
|
|
{
|
|
|
|
ItemStack item = *it;
|
|
|
|
item.mType = ItemStack::Type_Barter;
|
|
|
|
mItems.push_back(item);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|