2012-03-10 11:43:48 +00:00
|
|
|
#ifndef GAME_MWWORLD_INVENTORYSTORE_H
|
|
|
|
#define GAME_MWWORLD_INVENTORYSTORE_H
|
|
|
|
|
|
|
|
#include "containerstore.hpp"
|
|
|
|
|
2012-05-18 13:48:55 +00:00
|
|
|
#include "../mwmechanics/magiceffects.hpp"
|
|
|
|
|
2012-03-31 15:26:15 +00:00
|
|
|
namespace MWMechanics
|
|
|
|
{
|
2012-08-13 17:53:54 +00:00
|
|
|
class NpcStats;
|
2012-03-31 15:26:15 +00:00
|
|
|
}
|
|
|
|
|
2012-03-10 11:43:48 +00:00
|
|
|
namespace MWWorld
|
|
|
|
{
|
2013-11-15 01:08:36 +00:00
|
|
|
class InventoryStoreListener
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* Fired when items are equipped or unequipped
|
|
|
|
*/
|
|
|
|
virtual void equipmentChanged () {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param effect
|
|
|
|
* @param isNew Is this effect new (e.g. the item for it was just now manually equipped)
|
|
|
|
* or was it loaded from a savegame / initial game state? \n
|
|
|
|
* If it isn't new, non-looping VFX should not be played.
|
|
|
|
* @param playSound Play effect sound?
|
|
|
|
*/
|
|
|
|
virtual void permanentEffectAdded (const ESM::MagicEffect *magicEffect, bool isNew, bool playSound) {}
|
|
|
|
|
|
|
|
};
|
|
|
|
|
2012-03-10 11:43:48 +00:00
|
|
|
///< \brief Variant of the ContainerStore for NPCs
|
|
|
|
class InventoryStore : public ContainerStore
|
|
|
|
{
|
2012-03-13 12:31:11 +00:00
|
|
|
public:
|
2012-03-10 11:43:48 +00:00
|
|
|
|
2012-03-13 12:31:11 +00:00
|
|
|
static const int Slot_Helmet = 0;
|
|
|
|
static const int Slot_Cuirass = 1;
|
|
|
|
static const int Slot_Greaves = 2;
|
|
|
|
static const int Slot_LeftPauldron = 3;
|
|
|
|
static const int Slot_RightPauldron = 4;
|
|
|
|
static const int Slot_LeftGauntlet = 5;
|
|
|
|
static const int Slot_RightGauntlet = 6;
|
|
|
|
static const int Slot_Boots = 7;
|
|
|
|
static const int Slot_Shirt = 8;
|
|
|
|
static const int Slot_Pants = 9;
|
|
|
|
static const int Slot_Skirt = 10;
|
|
|
|
static const int Slot_Robe = 11;
|
|
|
|
static const int Slot_LeftRing = 12;
|
|
|
|
static const int Slot_RightRing = 13;
|
|
|
|
static const int Slot_Amulet = 14;
|
|
|
|
static const int Slot_Belt = 15;
|
|
|
|
static const int Slot_CarriedRight = 16;
|
|
|
|
static const int Slot_CarriedLeft = 17;
|
|
|
|
static const int Slot_Ammunition = 18;
|
2012-03-10 11:43:48 +00:00
|
|
|
|
2012-03-13 12:31:11 +00:00
|
|
|
static const int Slots = 19;
|
|
|
|
|
|
|
|
static const int Slot_NoSlot = -1;
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
2013-11-13 17:51:28 +00:00
|
|
|
MWMechanics::MagicEffects mMagicEffects;
|
|
|
|
|
2013-11-15 01:08:36 +00:00
|
|
|
InventoryStoreListener* mListener;
|
|
|
|
|
2013-11-13 17:51:28 +00:00
|
|
|
// Enables updates of magic effects and actor model whenever items are equipped or unequipped.
|
|
|
|
// This is disabled during autoequip to avoid excessive updates
|
|
|
|
bool mUpdatesEnabled;
|
|
|
|
|
2013-11-14 13:41:10 +00:00
|
|
|
bool mFirstAutoEquip;
|
|
|
|
|
2013-11-13 17:51:28 +00:00
|
|
|
// Vanilla allows permanent effects with a random magnitude, so it needs to be stored here.
|
|
|
|
// We also need this to only play sounds and particle effects when the item is equipped, rather than on every update.
|
|
|
|
typedef std::map<std::string, std::vector<float> > TEffectMagnitudes;
|
|
|
|
TEffectMagnitudes mPermanentMagicEffectMagnitudes;
|
2012-05-18 13:48:55 +00:00
|
|
|
|
2012-03-31 15:26:15 +00:00
|
|
|
typedef std::vector<ContainerStoreIterator> TSlots;
|
|
|
|
|
2013-11-13 17:51:28 +00:00
|
|
|
TSlots mSlots;
|
2012-03-13 12:31:11 +00:00
|
|
|
|
2012-05-29 10:35:03 +00:00
|
|
|
// selected magic item (for using enchantments of type "Cast once" or "Cast when used")
|
|
|
|
ContainerStoreIterator mSelectedEnchantItem;
|
|
|
|
|
2012-03-13 12:31:11 +00:00
|
|
|
void copySlots (const InventoryStore& store);
|
|
|
|
|
2013-11-12 22:23:19 +00:00
|
|
|
void initSlots (TSlots& slots_);
|
2012-03-31 15:26:15 +00:00
|
|
|
|
2013-11-15 01:08:36 +00:00
|
|
|
void updateMagicEffects();
|
2013-10-25 20:16:52 +00:00
|
|
|
|
2013-11-15 01:08:36 +00:00
|
|
|
void fireEquipmentChangedEvent();
|
2013-11-13 17:51:28 +00:00
|
|
|
|
2012-03-13 12:31:11 +00:00
|
|
|
public:
|
|
|
|
|
|
|
|
InventoryStore();
|
|
|
|
|
|
|
|
InventoryStore (const InventoryStore& store);
|
|
|
|
|
|
|
|
InventoryStore& operator= (const InventoryStore& store);
|
|
|
|
|
2013-08-07 12:45:23 +00:00
|
|
|
virtual ContainerStoreIterator add (const Ptr& itemPtr, const Ptr& actorPtr);
|
|
|
|
///< Add the item pointed to by \a ptr to this container. (Stacks automatically if needed)
|
|
|
|
/// Auto-equip items if specific conditions are fulfilled (see the implementation).
|
|
|
|
///
|
|
|
|
/// \note The item pointed to is not required to exist beyond this function call.
|
|
|
|
///
|
|
|
|
/// \attention Do not add items to an existing stack by increasing the count instead of
|
|
|
|
/// calling this function!
|
|
|
|
///
|
|
|
|
/// @return if stacking happened, return iterator to the item that was stacked against, otherwise iterator to the newly inserted item.
|
|
|
|
|
2013-08-13 00:06:46 +00:00
|
|
|
void equip (int slot, const ContainerStoreIterator& iterator, const Ptr& actor);
|
2012-05-29 10:35:03 +00:00
|
|
|
///< \note \a iterator can be an end-iterator
|
|
|
|
|
|
|
|
void setSelectedEnchantItem(const ContainerStoreIterator& iterator);
|
|
|
|
///< set the selected magic item (for using enchantments of type "Cast once" or "Cast when used")
|
|
|
|
/// \note to unset the selected item, call this method with end() iterator
|
|
|
|
|
|
|
|
ContainerStoreIterator getSelectedEnchantItem();
|
|
|
|
///< @return selected magic item (for using enchantments of type "Cast once" or "Cast when used")
|
|
|
|
/// \note if no item selected, return end() iterator
|
2012-03-13 12:31:11 +00:00
|
|
|
|
|
|
|
ContainerStoreIterator getSlot (int slot);
|
2012-03-31 15:26:15 +00:00
|
|
|
|
2013-08-06 12:55:08 +00:00
|
|
|
void unequipAll(const MWWorld::Ptr& actor);
|
2013-08-06 09:20:51 +00:00
|
|
|
///< Unequip all currently equipped items.
|
|
|
|
|
2013-04-05 13:42:05 +00:00
|
|
|
void autoEquip (const MWWorld::Ptr& npc);
|
2012-03-31 15:26:15 +00:00
|
|
|
///< Auto equip items according to stats and item value.
|
2012-05-13 12:58:38 +00:00
|
|
|
|
2013-11-13 17:51:28 +00:00
|
|
|
const MWMechanics::MagicEffects& getMagicEffects() const;
|
2012-05-18 13:48:55 +00:00
|
|
|
///< Return magic effects from worn items.
|
|
|
|
|
|
|
|
virtual void flagAsModified();
|
|
|
|
///< \attention This function is internal to the world model and should not be called from
|
|
|
|
/// outside.
|
|
|
|
|
2013-11-12 22:12:56 +00:00
|
|
|
virtual bool stacks (const Ptr& ptr1, const Ptr& ptr2);
|
2012-05-13 12:58:38 +00:00
|
|
|
///< @return true if the two specified objects can stack with each other
|
|
|
|
|
2013-08-12 23:19:33 +00:00
|
|
|
virtual int remove(const Ptr& item, int count, const Ptr& actor);
|
|
|
|
///< Remove \a count item(s) designated by \a item from this inventory.
|
|
|
|
///
|
|
|
|
/// @return the number of items actually removed
|
2013-10-31 23:54:54 +00:00
|
|
|
|
2013-11-09 01:47:11 +00:00
|
|
|
ContainerStoreIterator unequipSlot(int slot, const Ptr& actor, bool restack = true);
|
2013-11-01 00:01:55 +00:00
|
|
|
///< Unequip \a slot.
|
|
|
|
///
|
|
|
|
/// @return an iterator to the item that was previously in the slot
|
2013-11-09 01:47:11 +00:00
|
|
|
/// (if \a restack is true, the item can be re-stacked so its count
|
|
|
|
/// may differ from when it was equipped).
|
2013-10-31 23:21:15 +00:00
|
|
|
|
|
|
|
ContainerStoreIterator unequipItem(const Ptr& item, const Ptr& actor);
|
|
|
|
///< Unequip an item identified by its Ptr. An exception is thrown
|
|
|
|
/// if the item is not currently equipped.
|
|
|
|
///
|
|
|
|
/// @return an iterator to the item that was previously in the slot
|
|
|
|
/// (it can be re-stacked so its count may be different than when it
|
|
|
|
/// was equipped).
|
2013-11-15 01:08:36 +00:00
|
|
|
|
|
|
|
void setListener (InventoryStoreListener* listener);
|
|
|
|
///< Set a listener for various events, see \a InventoryStoreListener
|
2012-03-10 11:43:48 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|