1
0
Fork 1
mirror of https://github.com/TES3MP/openmw-tes3mp.git synced 2025-01-19 22:53:50 +00:00
openmw-tes3mp/apps/openmw/mwgui/inventorywindow.cpp

560 lines
21 KiB
C++
Raw Normal View History

2012-04-21 08:51:01 +00:00
#include "inventorywindow.hpp"
2013-05-11 16:38:27 +00:00
#include <stdexcept>
#include <boost/lexical_cast.hpp>
#include "../mwbase/world.hpp"
#include "../mwbase/environment.hpp"
#include "../mwbase/soundmanager.hpp"
#include "../mwbase/windowmanager.hpp"
2014-01-07 18:49:16 +00:00
#include "../mwbase/mechanicsmanager.hpp"
#include "../mwworld/inventorystore.hpp"
2013-05-11 16:38:27 +00:00
#include "../mwworld/class.hpp"
#include "../mwworld/action.hpp"
#include "../mwscript/interpretercontext.hpp"
#include "../mwbase/scriptmanager.hpp"
#include "bookwindow.hpp"
#include "scrollwindow.hpp"
2012-05-29 10:35:03 +00:00
#include "spellwindow.hpp"
2013-05-11 16:38:27 +00:00
#include "itemview.hpp"
#include "inventoryitemmodel.hpp"
#include "sortfilteritemmodel.hpp"
#include "tradeitemmodel.hpp"
#include "countdialog.hpp"
#include "tradewindow.hpp"
#include "container.hpp"
2012-04-21 08:51:01 +00:00
namespace MWGui
{
InventoryWindow::InventoryWindow(DragAndDrop* dragAndDrop)
2013-05-11 16:38:27 +00:00
: WindowPinnableBase("openmw_inventory_window.layout")
, mTrading(false)
2012-09-14 15:10:10 +00:00
, mLastXSize(0)
, mLastYSize(0)
, mPreview(MWBase::Environment::get().getWorld ()->getPlayerPtr())
, mPreviewDirty(true)
2013-05-11 16:38:27 +00:00
, mDragAndDrop(dragAndDrop)
2013-07-31 16:46:32 +00:00
, mSelectedItem(-1)
2013-07-31 19:40:29 +00:00
, mGuiMode(GM_Inventory)
2012-04-21 08:51:01 +00:00
{
static_cast<MyGUI::Window*>(mMainWidget)->eventWindowChangeCoord += MyGUI::newDelegate(this, &InventoryWindow::onWindowResize);
2012-05-12 19:28:04 +00:00
getWidget(mAvatar, "Avatar");
2012-09-12 22:54:32 +00:00
getWidget(mAvatarImage, "AvatarImage");
2012-05-12 19:28:04 +00:00
getWidget(mEncumbranceBar, "EncumbranceBar");
getWidget(mFilterAll, "AllButton");
getWidget(mFilterWeapon, "WeaponButton");
getWidget(mFilterApparel, "ApparelButton");
getWidget(mFilterMagic, "MagicButton");
getWidget(mFilterMisc, "MiscButton");
getWidget(mLeftPane, "LeftPane");
getWidget(mRightPane, "RightPane");
2013-03-16 21:53:33 +00:00
getWidget(mArmorRating, "ArmorRating");
2012-05-12 19:44:33 +00:00
mAvatarImage->eventMouseButtonClick += MyGUI::newDelegate(this, &InventoryWindow::onAvatarClicked);
2013-05-11 16:38:27 +00:00
getWidget(mItemView, "ItemView");
mItemView->eventItemClicked += MyGUI::newDelegate(this, &InventoryWindow::onItemSelected);
mItemView->eventBackgroundClicked += MyGUI::newDelegate(this, &InventoryWindow::onBackgroundSelected);
2012-05-12 20:44:12 +00:00
2012-05-12 19:44:33 +00:00
mFilterAll->eventMouseButtonClick += MyGUI::newDelegate(this, &InventoryWindow::onFilterChanged);
mFilterWeapon->eventMouseButtonClick += MyGUI::newDelegate(this, &InventoryWindow::onFilterChanged);
mFilterApparel->eventMouseButtonClick += MyGUI::newDelegate(this, &InventoryWindow::onFilterChanged);
mFilterMagic->eventMouseButtonClick += MyGUI::newDelegate(this, &InventoryWindow::onFilterChanged);
mFilterMisc->eventMouseButtonClick += MyGUI::newDelegate(this, &InventoryWindow::onFilterChanged);
mFilterAll->setStateSelected(true);
2013-12-03 17:42:35 +00:00
setGuiMode(mGuiMode);
adjustPanes();
}
void InventoryWindow::adjustPanes()
{
const float aspect = 0.5; // fixed aspect ratio for the avatar image
float leftPaneWidth = (mMainWidget->getSize().height-44-mArmorRating->getHeight()) * aspect;
mLeftPane->setSize( leftPaneWidth, mMainWidget->getSize().height-44 );
mRightPane->setCoord( mLeftPane->getPosition().left + leftPaneWidth + 4,
2013-12-03 17:42:35 +00:00
mRightPane->getPosition().top,
mMainWidget->getSize().width - 12 - leftPaneWidth - 15,
2013-12-03 17:42:35 +00:00
mMainWidget->getSize().height-44 );
2013-05-11 16:38:27 +00:00
}
2013-05-15 15:54:18 +00:00
void InventoryWindow::updatePlayer()
{
mPtr = MWBase::Environment::get().getWorld ()->getPlayerPtr();
2013-05-15 15:54:18 +00:00
mTradeModel = new TradeItemModel(new InventoryItemModel(mPtr), MWWorld::Ptr());
mSortModel = new SortFilterItemModel(mTradeModel);
mItemView->setModel(mSortModel);
mPreview = MWRender::InventoryPreview(mPtr);
mPreview.setup();
}
2013-07-31 19:40:29 +00:00
void InventoryWindow::setGuiMode(GuiMode mode)
{
2013-12-03 17:42:35 +00:00
std::string setting = "inventory";
2013-07-31 19:40:29 +00:00
mGuiMode = mode;
switch(mode) {
case GM_Container:
setPinButtonVisible(false);
2013-12-03 17:42:35 +00:00
setting += " container";
2013-07-31 19:40:29 +00:00
break;
case GM_Companion:
setPinButtonVisible(false);
2013-12-03 17:42:35 +00:00
setting += " companion";
2013-07-31 19:40:29 +00:00
break;
case GM_Barter:
setPinButtonVisible(false);
2013-12-03 17:42:35 +00:00
setting += " barter";
2013-07-31 19:40:29 +00:00
break;
case GM_Inventory:
default:
setPinButtonVisible(true);
2013-07-31 19:40:29 +00:00
break;
}
2013-12-03 17:42:35 +00:00
MyGUI::IntSize viewSize = MyGUI::RenderManager::getInstance().getViewSize();
MyGUI::IntPoint pos (Settings::Manager::getFloat(setting + " x", "Windows") * viewSize.width,
Settings::Manager::getFloat(setting + " y", "Windows") * viewSize.height);
MyGUI::IntSize size (Settings::Manager::getFloat(setting + " w", "Windows") * viewSize.width,
Settings::Manager::getFloat(setting + " h", "Windows") * viewSize.height);
if (size.width != mMainWidget->getWidth() || size.height != mMainWidget->getHeight())
mPreviewDirty = true;
2013-12-03 17:42:35 +00:00
mMainWidget->setPosition(pos);
mMainWidget->setSize(size);
adjustPanes();
2013-07-31 19:40:29 +00:00
}
2013-05-11 16:38:27 +00:00
TradeItemModel* InventoryWindow::getTradeModel()
{
return mTradeModel;
}
ItemModel* InventoryWindow::getModel()
{
return mTradeModel;
}
void InventoryWindow::onBackgroundSelected()
{
if (mDragAndDrop->mIsOnDragAndDrop)
mDragAndDrop->drop(mTradeModel, mItemView);
}
void InventoryWindow::onItemSelected (int index)
{
onItemSelectedFromSourceModel (mSortModel->mapToSource(index));
}
void InventoryWindow::onItemSelectedFromSourceModel (int index)
{
if (mDragAndDrop->mIsOnDragAndDrop)
{
mDragAndDrop->drop(mTradeModel, mItemView);
return;
}
const ItemStack& item = mTradeModel->getItem(index);
MWWorld::Ptr object = item.mBase;
int count = item.mCount;
// Bound items may not be moved
if (item.mBase.getCellRef().mRefID.size() > 6
&& item.mBase.getCellRef().mRefID.substr(0,6) == "bound_")
{
MWBase::Environment::get().getWindowManager()->messageBox("#{sBarterDialog12}");
return;
}
if (item.mType == ItemStack::Type_Equipped)
{
MWWorld::InventoryStore& invStore = MWWorld::Class::get(mPtr).getInventoryStore(mPtr);
MWWorld::Ptr newStack = *invStore.unequipItem(item.mBase, mPtr);
// The unequipped item was re-stacked. We have to update the index
// since the item pointed does not exist anymore.
if (item.mBase != newStack)
{
// newIndex will store the index of the ItemStack the item was stacked on
int newIndex = -1;
for (size_t i=0; i < mTradeModel->getItemCount(); ++i)
{
if (mTradeModel->getItem(i).mBase == newStack)
{
newIndex = i;
break;
}
}
if (newIndex == -1)
throw std::runtime_error("Can't find restacked item");
index = newIndex;
object = mTradeModel->getItem(index).mBase;
}
}
2013-05-11 16:38:27 +00:00
bool shift = MyGUI::InputManager::getInstance().isShiftPressed();
if (MyGUI::InputManager::getInstance().isControlPressed())
count = 1;
if (mTrading)
{
// check if merchant accepts item
int services = MWBase::Environment::get().getWindowManager()->getTradeWindow()->getMerchantServices();
if (!MWWorld::Class::get(object).canSell(object, services))
{
MWBase::Environment::get().getWindowManager()->
messageBox("#{sBarterDialog4}");
return;
}
}
if (count > 1 && !shift)
{
CountDialog* dialog = MWBase::Environment::get().getWindowManager()->getCountDialog();
std::string message = mTrading ? "#{sQuanityMenuMessage01}" : "#{sTake}";
dialog->open(MWWorld::Class::get(object).getName(object), message, count);
dialog->eventOkClicked.clear();
if (mTrading)
dialog->eventOkClicked += MyGUI::newDelegate(this, &InventoryWindow::sellItem);
else
dialog->eventOkClicked += MyGUI::newDelegate(this, &InventoryWindow::dragItem);
mSelectedItem = index;
}
else
{
mSelectedItem = index;
if (mTrading)
sellItem (NULL, count);
else
dragItem (NULL, count);
}
// item might have been unequipped
notifyContentChanged();
}
void InventoryWindow::dragItem(MyGUI::Widget* sender, int count)
{
mDragAndDrop->startDrag(mSelectedItem, mSortModel, mTradeModel, mItemView, count);
}
void InventoryWindow::sellItem(MyGUI::Widget* sender, int count)
{
const ItemStack& item = mTradeModel->getItem(mSelectedItem);
std::string sound = MWWorld::Class::get(item.mBase).getDownSoundId(item.mBase);
MWBase::Environment::get().getSoundManager()->playSound (sound, 1.0, 1.0);
if (item.mType == ItemStack::Type_Barter)
{
// this was an item borrowed to us by the merchant
MWBase::Environment::get().getWindowManager()->getTradeWindow()->returnItem(mSelectedItem, count);
mTradeModel->returnItemBorrowedToUs(mSelectedItem, count);
}
else
{
// borrow item to the merchant
MWBase::Environment::get().getWindowManager()->getTradeWindow()->borrowItem(mSelectedItem, count);
mTradeModel->borrowItemFromUs(mSelectedItem, count);
}
mItemView->update();
}
void InventoryWindow::updateItemView()
{
mItemView->update();
mPreviewDirty = true;
}
2012-05-12 19:28:04 +00:00
void InventoryWindow::open()
{
mPtr = MWBase::Environment::get().getWorld()->getPlayerPtr();
2012-05-15 20:45:46 +00:00
updateEncumbranceBar();
2013-05-11 16:38:27 +00:00
mItemView->update();
2013-05-11 16:38:27 +00:00
notifyContentChanged();
2013-12-03 17:42:35 +00:00
adjustPanes();
}
void InventoryWindow::onWindowResize(MyGUI::Window* _sender)
{
2013-12-03 17:42:35 +00:00
adjustPanes();
std::string setting = "inventory";
2013-07-31 19:40:29 +00:00
switch(mGuiMode) {
case GM_Container:
2013-12-03 17:42:35 +00:00
setting += " container";
2013-07-31 19:40:29 +00:00
break;
case GM_Companion:
2013-12-03 17:42:35 +00:00
setting += " companion";
2013-07-31 19:40:29 +00:00
break;
case GM_Barter:
2013-12-03 17:42:35 +00:00
setting += " barter";
2013-07-31 19:40:29 +00:00
break;
default:
2013-12-03 17:42:35 +00:00
break;
2013-07-31 19:40:29 +00:00
}
2013-12-03 17:42:35 +00:00
MyGUI::IntSize viewSize = MyGUI::RenderManager::getInstance().getViewSize();
float x = _sender->getPosition().left / float(viewSize.width);
float y = _sender->getPosition().top / float(viewSize.height);
float w = _sender->getSize().width / float(viewSize.width);
float h = _sender->getSize().height / float(viewSize.height);
Settings::Manager::setFloat(setting + " x", "Windows", x);
Settings::Manager::setFloat(setting + " y", "Windows", y);
Settings::Manager::setFloat(setting + " w", "Windows", w);
Settings::Manager::setFloat(setting + " h", "Windows", h);
2012-09-14 15:10:10 +00:00
if (mMainWidget->getSize().width != mLastXSize || mMainWidget->getSize().height != mLastYSize)
{
mLastXSize = mMainWidget->getSize().width;
mLastYSize = mMainWidget->getSize().height;
2013-05-11 16:38:27 +00:00
mPreviewDirty = true;
2012-09-14 15:10:10 +00:00
}
}
2012-05-12 19:44:33 +00:00
void InventoryWindow::onFilterChanged(MyGUI::Widget* _sender)
{
if (_sender == mFilterAll)
2013-05-11 16:38:27 +00:00
mSortModel->setCategory(SortFilterItemModel::Category_All);
2012-05-12 19:44:33 +00:00
else if (_sender == mFilterWeapon)
2013-05-11 16:38:27 +00:00
mSortModel->setCategory(SortFilterItemModel::Category_Weapon);
2012-05-12 19:44:33 +00:00
else if (_sender == mFilterApparel)
2013-05-11 16:38:27 +00:00
mSortModel->setCategory(SortFilterItemModel::Category_Apparel);
2012-05-12 19:44:33 +00:00
else if (_sender == mFilterMagic)
2013-05-11 16:38:27 +00:00
mSortModel->setCategory(SortFilterItemModel::Category_Magic);
2012-05-12 19:44:33 +00:00
else if (_sender == mFilterMisc)
2013-05-11 16:38:27 +00:00
mSortModel->setCategory(SortFilterItemModel::Category_Misc);
mFilterAll->setStateSelected(false);
mFilterWeapon->setStateSelected(false);
mFilterApparel->setStateSelected(false);
mFilterMagic->setStateSelected(false);
mFilterMisc->setStateSelected(false);
2013-05-11 16:38:27 +00:00
mItemView->update();
static_cast<MyGUI::Button*>(_sender)->setStateSelected(true);
2012-05-12 19:44:33 +00:00
}
2012-05-12 20:44:12 +00:00
void InventoryWindow::onPinToggled()
{
MWBase::Environment::get().getWindowManager()->setWeaponVisibility(!mPinned);
2012-05-12 20:44:12 +00:00
}
void InventoryWindow::useItem(const MWWorld::Ptr &ptr)
{
const std::string& script = ptr.getClass().getScript(ptr);
// If the item has a script, set its OnPcEquip to 1
if (!script.empty()
// Another morrowind oddity: when an item has skipped equipping and pcskipequip is reset to 0 afterwards,
// the next time it is equipped will work normally, but will not set onpcequip
&& (ptr != mSkippedToEquip || ptr.getRefData().getLocals().getIntVar(script, "pcskipequip") == 1))
ptr.getRefData().getLocals().setVarByInt(script, "onpcequip", 1);
// Give the script a chance to run once before we do anything else
// this is important when setting pcskipequip as a reaction to onpcequip being set (bk_treasuryreport does this)
if (!script.empty())
{
MWScript::InterpreterContext interpreterContext (&ptr.getRefData().getLocals(), ptr);
MWBase::Environment::get().getScriptManager()->run (script, interpreterContext);
}
if (script.empty() || ptr.getRefData().getLocals().getIntVar(script, "pcskipequip") == 0)
{
boost::shared_ptr<MWWorld::Action> action = MWWorld::Class::get(ptr).use(ptr);
action->execute (MWBase::Environment::get().getWorld()->getPlayerPtr());
// this is necessary for books/scrolls: if they are already in the player's inventory,
// the "Take" button should not be visible.
// NOTE: the take button is "reset" when the window opens, so we can safely do the following
// without screwing up future book windows
MWBase::Environment::get().getWindowManager()->getBookWindow()->setTakeButtonShow(false);
MWBase::Environment::get().getWindowManager()->getScrollWindow()->setTakeButtonShow(false);
mSkippedToEquip = MWWorld::Ptr();
}
else
mSkippedToEquip = ptr;
mItemView->update();
notifyContentChanged();
}
void InventoryWindow::onAvatarClicked(MyGUI::Widget* _sender)
{
if (mDragAndDrop->mIsOnDragAndDrop)
{
2013-05-11 16:38:27 +00:00
MWWorld::Ptr ptr = mDragAndDrop->mItem.mBase;
mDragAndDrop->finish();
2013-05-11 16:38:27 +00:00
if (mDragAndDrop->mSourceModel != mTradeModel)
{
// add item to the player's inventory
MWWorld::ContainerStore& invStore = MWWorld::Class::get(mPtr).getContainerStore(mPtr);
MWWorld::ContainerStoreIterator it = invStore.begin();
it = invStore.add(ptr, mDragAndDrop->mDraggedCount, mPtr);
2013-05-11 16:38:27 +00:00
mDragAndDrop->mSourceModel->removeItem(mDragAndDrop->mItem, mDragAndDrop->mDraggedCount);
ptr = *it;
}
useItem(ptr);
}
2012-09-14 12:34:18 +00:00
else
{
MyGUI::IntPoint mousePos = MyGUI::InputManager::getInstance ().getLastPressedPosition (MyGUI::MouseButton::Left);
MyGUI::IntPoint relPos = mousePos - mAvatarImage->getAbsolutePosition ();
int realX = int(float(relPos.left) / float(mAvatarImage->getSize().width) * 512.f );
int realY = int(float(relPos.top) / float(mAvatarImage->getSize().height) * 1024.f );
MWWorld::Ptr itemSelected = getAvatarSelectedItem (realX, realY);
2012-09-14 12:34:18 +00:00
if (itemSelected.isEmpty ())
return;
2013-05-11 16:38:27 +00:00
for (size_t i=0; i < mTradeModel->getItemCount (); ++i)
2012-09-14 12:34:18 +00:00
{
2013-05-11 16:38:27 +00:00
if (mTradeModel->getItem(i).mBase == itemSelected)
2012-09-14 12:34:18 +00:00
{
2013-05-11 16:38:27 +00:00
onItemSelectedFromSourceModel(i);
2012-09-14 12:34:18 +00:00
return;
}
}
2013-05-11 16:38:27 +00:00
throw std::runtime_error("Can't find clicked item");
2012-09-14 12:34:18 +00:00
}
}
MWWorld::Ptr InventoryWindow::getAvatarSelectedItem(int x, int y)
{
int slot = mPreview.getSlotSelected (x, y);
if (slot == -1)
return MWWorld::Ptr();
MWWorld::InventoryStore& invStore = MWWorld::Class::get(mPtr).getInventoryStore(mPtr);
if(invStore.getSlot(slot) != invStore.end())
{
MWWorld::Ptr item = *invStore.getSlot(slot);
// NOTE: Don't allow users to select WerewolfRobe objects in the inventory. Vanilla
// likely uses a hack like this since there's no other way to prevent it from being
// taken.
if(item.getCellRef().mRefID == "werewolfrobe")
return MWWorld::Ptr();
return item;
}
return MWWorld::Ptr();
}
2012-05-15 20:45:46 +00:00
void InventoryWindow::updateEncumbranceBar()
{
MWWorld::Ptr player = MWBase::Environment::get().getWorld()->getPlayerPtr();
2012-05-15 20:45:46 +00:00
float capacity = MWWorld::Class::get(player).getCapacity(player);
float encumbrance = MWWorld::Class::get(player).getEncumbrance(player);
2013-03-31 11:13:46 +00:00
mEncumbranceBar->setValue(encumbrance, capacity);
2012-05-15 20:45:46 +00:00
}
void InventoryWindow::onFrame()
2012-05-15 20:45:46 +00:00
{
if (!mMainWidget->getVisible())
return;
2012-05-15 20:45:46 +00:00
updateEncumbranceBar();
}
2013-05-11 16:38:27 +00:00
void InventoryWindow::setTrading(bool trading)
{
2013-05-11 16:38:27 +00:00
mTrading = trading;
}
2012-05-29 10:35:03 +00:00
void InventoryWindow::doRenderUpdate ()
{
if (mPreviewDirty)
{
mPreviewDirty = false;
MyGUI::IntSize size = mAvatarImage->getSize();
mPreview.update (size.width, size.height);
mAvatarImage->setImageTexture("CharacterPreview");
mAvatarImage->setImageCoord(MyGUI::IntCoord(0, 0, std::min(512, size.width), std::min(1024, size.height)));
mAvatarImage->setImageTile(MyGUI::IntSize(std::min(512, size.width), std::min(1024, size.height)));
mArmorRating->setCaptionWithReplacing ("#{sArmor}: "
+ boost::lexical_cast<std::string>(static_cast<int>(MWWorld::Class::get(mPtr).getArmorRating(mPtr))));
if (mArmorRating->getTextSize().width > mArmorRating->getSize().width)
mArmorRating->setCaptionWithReplacing (boost::lexical_cast<std::string>(static_cast<int>(MWWorld::Class::get(mPtr).getArmorRating(mPtr))));
}
}
2012-05-29 10:35:03 +00:00
void InventoryWindow::notifyContentChanged()
{
// update the spell window just in case new enchanted items were added to inventory
if (MWBase::Environment::get().getWindowManager()->getSpellWindow())
MWBase::Environment::get().getWindowManager()->getSpellWindow()->updateSpells();
mPreviewDirty = true;
}
void InventoryWindow::pickUpObject (MWWorld::Ptr object)
{
// make sure the object is of a type that can be picked up
std::string type = object.getTypeName();
if ( (type != typeid(ESM::Apparatus).name())
&& (type != typeid(ESM::Armor).name())
&& (type != typeid(ESM::Book).name())
&& (type != typeid(ESM::Clothing).name())
&& (type != typeid(ESM::Ingredient).name())
&& (type != typeid(ESM::Light).name())
&& (type != typeid(ESM::Miscellaneous).name())
2013-03-22 04:50:54 +00:00
&& (type != typeid(ESM::Lockpick).name())
&& (type != typeid(ESM::Probe).name())
&& (type != typeid(ESM::Repair).name())
&& (type != typeid(ESM::Weapon).name())
&& (type != typeid(ESM::Potion).name()))
return;
if (MWWorld::Class::get(object).getName(object) == "") // objects without name presented to user can never be picked up
return;
int count = object.getRefData().getCount();
// add to player inventory
// can't use ActionTake here because we need an MWWorld::Ptr to the newly inserted object
MWWorld::Ptr player = MWBase::Environment::get().getWorld()->getPlayerPtr();
MWWorld::Ptr newObject = *player.getClass().getContainerStore (player).add (object, object.getRefData().getCount(), player);
// remove from world
MWBase::Environment::get().getWorld()->deleteObject (object);
2013-05-11 16:38:27 +00:00
// get ModelIndex to the item
mTradeModel->update();
size_t i=0;
for (; i<mTradeModel->getItemCount(); ++i)
{
if (mTradeModel->getItem(i).mBase == newObject)
break;
}
if (i == mTradeModel->getItemCount())
throw std::runtime_error("Added item not found");
mDragAndDrop->startDrag(i, mSortModel, mTradeModel, mItemView, count);
2014-01-07 18:49:16 +00:00
MWBase::Environment::get().getMechanicsManager()->itemTaken(player, newObject, count);
2012-05-29 10:35:03 +00:00
}
2012-04-22 19:06:08 +00:00
}