2012-05-16 16:08:55 +02:00
|
|
|
#include "actionequip.hpp"
|
|
|
|
|
|
|
|
#include "../mwbase/environment.hpp"
|
2013-01-09 03:03:14 +00:00
|
|
|
#include "../mwbase/windowmanager.hpp"
|
2012-07-03 12:30:50 +02:00
|
|
|
|
2015-08-21 21:12:39 +12:00
|
|
|
#include "../mwmechanics/actorutil.hpp"
|
|
|
|
|
2013-01-31 00:34:16 +00:00
|
|
|
#include <components/compiler/locals.hpp>
|
2012-07-03 12:30:50 +02:00
|
|
|
|
|
|
|
#include "inventorystore.hpp"
|
|
|
|
#include "player.hpp"
|
|
|
|
#include "class.hpp"
|
2012-05-16 16:08:55 +02:00
|
|
|
|
|
|
|
namespace MWWorld
|
|
|
|
{
|
2018-07-09 19:31:40 +04:00
|
|
|
ActionEquip::ActionEquip (const MWWorld::Ptr& object, bool force)
|
|
|
|
: Action (false, object)
|
|
|
|
, mForce(force)
|
2012-05-16 16:08:55 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2012-07-27 12:00:10 +02:00
|
|
|
void ActionEquip::executeImp (const Ptr& actor)
|
2012-05-16 16:08:55 +02:00
|
|
|
{
|
2013-04-09 16:14:08 +02:00
|
|
|
MWWorld::Ptr object = getTarget();
|
2014-01-19 11:42:58 +01:00
|
|
|
MWWorld::InventoryStore& invStore = actor.getClass().getInventoryStore(actor);
|
2012-05-16 16:08:55 +02:00
|
|
|
|
2018-07-09 19:31:40 +04:00
|
|
|
if (object.getClass().hasItemHealth(object) && object.getCellRef().getCharge() == 0)
|
|
|
|
{
|
|
|
|
if (actor == MWMechanics::getPlayer())
|
|
|
|
MWBase::Environment::get().getWindowManager()->messageBox("#{sInventoryMessage1}");
|
2013-04-15 02:56:23 +02:00
|
|
|
|
2018-07-09 19:31:40 +04:00
|
|
|
return;
|
|
|
|
}
|
2013-04-15 02:56:23 +02:00
|
|
|
|
2018-07-09 19:31:40 +04:00
|
|
|
if (!mForce)
|
2013-04-09 19:46:27 +02:00
|
|
|
{
|
2018-07-09 19:31:40 +04:00
|
|
|
std::pair <int, std::string> result = object.getClass().canBeEquipped (object, actor);
|
|
|
|
|
|
|
|
// display error message if the player tried to equip something
|
|
|
|
if (!result.second.empty() && actor == MWMechanics::getPlayer())
|
|
|
|
MWBase::Environment::get().getWindowManager()->messageBox(result.second);
|
|
|
|
|
|
|
|
switch(result.first)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
return;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2013-04-09 19:46:27 +02:00
|
|
|
}
|
|
|
|
|
2012-05-16 16:08:55 +02:00
|
|
|
// slots that this item can be equipped in
|
2014-05-22 20:37:22 +02:00
|
|
|
std::pair<std::vector<int>, bool> slots_ = getTarget().getClass().getEquipmentSlots(getTarget());
|
2014-06-05 14:54:07 +02:00
|
|
|
if (slots_.first.empty())
|
|
|
|
return;
|
2012-05-16 16:08:55 +02:00
|
|
|
|
|
|
|
// retrieve ContainerStoreIterator to the item
|
|
|
|
MWWorld::ContainerStoreIterator it = invStore.begin();
|
|
|
|
for (; it != invStore.end(); ++it)
|
|
|
|
{
|
2013-04-09 16:14:08 +02:00
|
|
|
if (*it == object)
|
2012-05-16 16:08:55 +02:00
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-24 22:50:54 +01:00
|
|
|
if (it == invStore.end())
|
|
|
|
{
|
|
|
|
std::stringstream error;
|
|
|
|
error << "ActionEquip can't find item " << object.getCellRef().getRefId();
|
|
|
|
throw std::runtime_error(error.str());
|
|
|
|
}
|
2013-02-02 13:24:28 +01:00
|
|
|
|
2012-05-16 16:08:55 +02:00
|
|
|
// equip the item in the first free slot
|
2014-06-05 14:54:07 +02:00
|
|
|
std::vector<int>::const_iterator slot=slots_.first.begin();
|
|
|
|
for (;slot!=slots_.first.end(); ++slot)
|
2012-05-16 16:08:55 +02:00
|
|
|
{
|
2013-11-17 13:37:19 +01:00
|
|
|
// if the item is equipped already, nothing to do
|
|
|
|
if (invStore.getSlot(*slot) == it)
|
|
|
|
return;
|
2013-04-02 11:42:29 +02:00
|
|
|
|
2014-06-05 14:54:07 +02:00
|
|
|
if (invStore.getSlot(*slot) == invStore.end())
|
2012-05-16 16:08:55 +02:00
|
|
|
{
|
2014-06-05 14:54:07 +02:00
|
|
|
// slot is not occupied
|
2013-08-13 02:06:46 +02:00
|
|
|
invStore.equip(*slot, it, actor);
|
2012-05-16 16:08:55 +02:00
|
|
|
break;
|
|
|
|
}
|
2014-06-05 14:54:07 +02:00
|
|
|
}
|
2012-05-16 16:08:55 +02:00
|
|
|
|
2014-06-05 14:54:07 +02:00
|
|
|
// all slots are occupied -> cycle
|
|
|
|
// move all slots one towards begin(), then equip the item in the slot that is now free
|
|
|
|
if (slot == slots_.first.end())
|
|
|
|
{
|
2021-04-06 20:12:51 -05:00
|
|
|
ContainerStoreIterator enchItem = invStore.getSelectedEnchantItem();
|
2021-04-07 16:57:06 -05:00
|
|
|
bool reEquip = false;
|
2021-04-06 20:12:51 -05:00
|
|
|
for (slot = slots_.first.begin(); slot != slots_.first.end(); ++slot)
|
2012-05-16 16:08:55 +02:00
|
|
|
{
|
2019-12-14 21:30:46 +04:00
|
|
|
invStore.unequipSlot(*slot, actor, false);
|
2021-04-06 20:12:51 -05:00
|
|
|
if (slot + 1 != slots_.first.end())
|
|
|
|
{
|
|
|
|
invStore.equip(*slot, invStore.getSlot(*(slot + 1)), actor);
|
|
|
|
}
|
2014-06-05 14:54:07 +02:00
|
|
|
else
|
2021-04-06 20:12:51 -05:00
|
|
|
{
|
2014-06-05 14:54:07 +02:00
|
|
|
invStore.equip(*slot, it, actor);
|
2021-04-06 20:12:51 -05:00
|
|
|
}
|
|
|
|
|
2021-04-07 16:57:06 -05:00
|
|
|
//Fix for issue of selected enchated item getting remmoved on cycle
|
2021-04-06 20:12:51 -05:00
|
|
|
if (invStore.getSlot(*slot) == enchItem)
|
|
|
|
{
|
2021-04-07 16:57:06 -05:00
|
|
|
reEquip = true;
|
2021-04-06 20:12:51 -05:00
|
|
|
}
|
|
|
|
}
|
2021-04-07 16:57:06 -05:00
|
|
|
if (reEquip)
|
2021-04-06 20:12:51 -05:00
|
|
|
{
|
|
|
|
invStore.setSelectedEnchantItem(enchItem);
|
2012-05-16 16:08:55 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|