2013-05-11 18:38:27 +02:00
# include "inventoryitemmodel.hpp"
2017-09-11 14:49:55 +04:00
# include <sstream>
# include "../mwmechanics/actorutil.hpp"
# include "../mwmechanics/npcstats.hpp"
2013-05-11 18:38:27 +02:00
# include "../mwworld/containerstore.hpp"
# include "../mwworld/class.hpp"
# include "../mwworld/inventorystore.hpp"
2017-10-04 21:25:22 +04:00
# include "../mwbase/environment.hpp"
# include "../mwbase/mechanicsmanager.hpp"
2013-05-11 18:38:27 +02:00
namespace MWGui
{
InventoryItemModel : : InventoryItemModel ( const MWWorld : : Ptr & actor )
: mActor ( actor )
{
}
ItemStack InventoryItemModel : : 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 InventoryItemModel : : getItemCount ( )
{
return mItems . size ( ) ;
}
2021-08-15 19:50:28 +02:00
ItemModel : : ModelIndex InventoryItemModel : : getIndex ( const ItemStack & item )
2013-05-11 18:38:27 +02:00
{
size_t i = 0 ;
2019-03-02 13:27:59 +04:00
for ( ItemStack & itemStack : mItems )
2013-05-11 18:38:27 +02:00
{
2019-03-02 13:27:59 +04:00
if ( itemStack = = item )
2013-05-11 18:38:27 +02:00
return i ;
+ + i ;
}
return - 1 ;
}
2019-10-28 01:58:23 +03:00
MWWorld : : Ptr InventoryItemModel : : copyItem ( const ItemStack & item , size_t count , bool allowAutoEquip )
2013-05-11 18:38:27 +02:00
{
2013-08-25 17:40:08 +02:00
if ( item . mBase . getContainerStore ( ) = = & mActor . getClass ( ) . getContainerStore ( mActor ) )
throw std : : runtime_error ( " Item to copy needs to be from a different container! " ) ;
2019-10-28 01:58:23 +03:00
return * mActor . getClass ( ) . getContainerStore ( mActor ) . add ( item . mBase , count , mActor , allowAutoEquip ) ;
2013-05-11 18:38:27 +02:00
}
void InventoryItemModel : : removeItem ( const ItemStack & item , size_t count )
{
2017-09-11 14:49:55 +04:00
int removed = 0 ;
// Re-equipping makes sense only if a target has inventory
if ( mActor . getClass ( ) . hasInventoryStore ( mActor ) )
{
MWWorld : : InventoryStore & store = mActor . getClass ( ) . getInventoryStore ( mActor ) ;
removed = store . remove ( item . mBase , count , mActor , true ) ;
}
else
{
MWWorld : : ContainerStore & store = mActor . getClass ( ) . getContainerStore ( mActor ) ;
removed = store . remove ( item . mBase , count , mActor ) ;
}
std : : stringstream error ;
2013-05-11 18:38:27 +02:00
2013-08-13 01:19:33 +02:00
if ( removed = = 0 )
2017-09-11 14:49:55 +04:00
{
error < < " Item ' " < < item . mBase . getCellRef ( ) . getRefId ( ) < < " ' was not found in container store to remove " ;
throw std : : runtime_error ( error . str ( ) ) ;
}
2013-11-12 22:58:56 +01:00
else if ( removed < static_cast < int > ( count ) )
2017-09-11 14:49:55 +04:00
{
error < < " Not enough items ' " < < item . mBase . getCellRef ( ) . getRefId ( ) < < " ' in the stack to remove ( " < < static_cast < int > ( count ) < < " requested, " < < removed < < " found) " ;
throw std : : runtime_error ( error . str ( ) ) ;
}
2013-05-11 18:38:27 +02:00
}
2014-05-15 04:41:31 +02:00
MWWorld : : Ptr InventoryItemModel : : moveItem ( const ItemStack & item , size_t count , ItemModel * otherModel )
2014-04-27 05:40:07 +02:00
{
2014-09-10 23:38:38 -04:00
// Can't move conjured items: This is a general fix that also takes care of issues with taking conjured items via the 'Take All' button.
2014-09-13 18:59:13 -04:00
if ( item . mFlags & ItemStack : : Flag_Bound )
2014-09-10 23:38:38 -04:00
return MWWorld : : Ptr ( ) ;
2019-10-05 16:47:41 +04:00
MWWorld : : Ptr ret = otherModel - > copyItem ( item , count ) ;
2014-04-27 05:40:07 +02:00
removeItem ( item , count ) ;
2014-05-15 04:41:31 +02:00
return ret ;
2014-04-27 05:40:07 +02:00
}
2013-05-11 18:38:27 +02:00
void InventoryItemModel : : update ( )
{
2014-05-22 20:37:22 +02:00
MWWorld : : ContainerStore & store = mActor . getClass ( ) . getContainerStore ( mActor ) ;
2013-05-11 18:38:27 +02:00
mItems . clear ( ) ;
for ( MWWorld : : ContainerStoreIterator it = store . begin ( ) ; it ! = store . end ( ) ; + + it )
{
2013-08-08 16:05:26 -07:00
MWWorld : : Ptr item = * it ;
2016-09-24 18:01:31 +02:00
if ( ! item . getClass ( ) . showsInInventory ( item ) )
2013-08-08 16:05:26 -07:00
continue ;
ItemStack newItem ( item , this , item . getRefData ( ) . getCount ( ) ) ;
2013-05-11 18:38:27 +02:00
2014-01-19 11:42:58 +01:00
if ( mActor . getClass ( ) . hasInventoryStore ( mActor ) )
2013-05-11 18:38:27 +02:00
{
2016-10-31 00:23:51 +09:00
MWWorld : : InventoryStore & invStore = mActor . getClass ( ) . getInventoryStore ( mActor ) ;
if ( invStore . isEquipped ( newItem . mBase ) )
2014-12-15 13:47:34 +01:00
newItem . mType = ItemStack : : Type_Equipped ;
2013-05-11 18:38:27 +02:00
}
mItems . push_back ( newItem ) ;
}
}
2017-10-04 23:26:06 +04:00
bool InventoryItemModel : : onTakeItem ( const MWWorld : : Ptr & item , int count )
2017-10-04 21:25:22 +04:00
{
// Looting a dead corpse is considered OK
if ( mActor . getClass ( ) . isActor ( ) & & mActor . getClass ( ) . getCreatureStats ( mActor ) . isDead ( ) )
return true ;
2017-10-04 22:37:08 +04:00
2017-10-04 21:25:22 +04:00
MWWorld : : Ptr player = MWMechanics : : getPlayer ( ) ;
MWBase : : Environment : : get ( ) . getMechanicsManager ( ) - > itemTaken ( player , item , mActor , count ) ;
return true ;
}
2021-07-15 22:31:26 +02:00
bool InventoryItemModel : : usesContainer ( const MWWorld : : Ptr & container )
{
return mActor = = container ;
}
2013-05-11 18:38:27 +02:00
}