mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-18 04:10:06 +00:00
b5ec584be2
Static const is only required to provide a reference or a pointer when it is not possible with default constructed temporary.
142 lines
4.7 KiB
C++
142 lines
4.7 KiB
C++
#include "pickpocketitemmodel.hpp"
|
|
|
|
#include <components/esm3/loadskil.hpp>
|
|
#include <components/misc/rng.hpp>
|
|
|
|
#include "../mwmechanics/actorutil.hpp"
|
|
#include "../mwmechanics/creaturestats.hpp"
|
|
#include "../mwmechanics/pickpocket.hpp"
|
|
|
|
#include "../mwworld/class.hpp"
|
|
|
|
#include "../mwbase/environment.hpp"
|
|
#include "../mwbase/mechanicsmanager.hpp"
|
|
#include "../mwbase/windowmanager.hpp"
|
|
#include "../mwbase/world.hpp"
|
|
|
|
namespace MWGui
|
|
{
|
|
|
|
PickpocketItemModel::PickpocketItemModel(
|
|
const MWWorld::Ptr& actor, std::unique_ptr<ItemModel> sourceModel, bool hideItems)
|
|
: mActor(actor)
|
|
, mPickpocketDetected(false)
|
|
{
|
|
MWWorld::Ptr player = MWMechanics::getPlayer();
|
|
mSourceModel = std::move(sourceModel);
|
|
float chance = player.getClass().getSkill(player, ESM::Skill::Sneak);
|
|
|
|
mSourceModel->update();
|
|
// build list of items that player is unable to find when attempts to pickpocket.
|
|
if (hideItems)
|
|
{
|
|
auto& prng = MWBase::Environment::get().getWorld()->getPrng();
|
|
for (size_t i = 0; i < mSourceModel->getItemCount(); ++i)
|
|
{
|
|
if (Misc::Rng::roll0to99(prng) > chance)
|
|
mHiddenItems.push_back(mSourceModel->getItem(i));
|
|
}
|
|
}
|
|
}
|
|
|
|
bool PickpocketItemModel::allowedToUseItems() const
|
|
{
|
|
return false;
|
|
}
|
|
|
|
ItemStack PickpocketItemModel::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 PickpocketItemModel::getItemCount()
|
|
{
|
|
return mItems.size();
|
|
}
|
|
|
|
void PickpocketItemModel::update()
|
|
{
|
|
mSourceModel->update();
|
|
mItems.clear();
|
|
for (size_t i = 0; i < mSourceModel->getItemCount(); ++i)
|
|
{
|
|
const ItemStack& item = mSourceModel->getItem(i);
|
|
|
|
// Bound items may not be stolen
|
|
if (item.mFlags & ItemStack::Flag_Bound)
|
|
continue;
|
|
|
|
if (std::find(mHiddenItems.begin(), mHiddenItems.end(), item) == mHiddenItems.end()
|
|
&& item.mType != ItemStack::Type_Equipped)
|
|
mItems.push_back(item);
|
|
}
|
|
}
|
|
|
|
void PickpocketItemModel::removeItem(const ItemStack& item, size_t count)
|
|
{
|
|
ProxyItemModel::removeItem(item, count);
|
|
}
|
|
|
|
bool PickpocketItemModel::onDropItem(const MWWorld::Ptr& item, int count)
|
|
{
|
|
// don't allow "reverse pickpocket" (it will be handled by scripts after 1.0)
|
|
return false;
|
|
}
|
|
|
|
void PickpocketItemModel::onClose()
|
|
{
|
|
// Make sure we were actually closed, rather than just temporarily hidden (e.g. console or main menu opened)
|
|
if (MWBase::Environment::get().getWindowManager()->containsMode(GM_Container)
|
|
// If it was already detected while taking an item, no need to check now
|
|
|| mPickpocketDetected)
|
|
return;
|
|
|
|
MWWorld::Ptr player = MWMechanics::getPlayer();
|
|
MWMechanics::Pickpocket pickpocket(player, mActor);
|
|
if (pickpocket.finish())
|
|
{
|
|
MWBase::Environment::get().getMechanicsManager()->commitCrime(
|
|
player, mActor, MWBase::MechanicsManager::OT_Pickpocket, ESM::RefId(), 0, true);
|
|
MWBase::Environment::get().getWindowManager()->removeGuiMode(MWGui::GM_Container);
|
|
mPickpocketDetected = true;
|
|
}
|
|
}
|
|
|
|
bool PickpocketItemModel::onTakeItem(const MWWorld::Ptr& item, int count)
|
|
{
|
|
if (mActor.getClass().getCreatureStats(mActor).getKnockedDown())
|
|
return mSourceModel->onTakeItem(item, count);
|
|
|
|
bool success = stealItem(item, count);
|
|
if (success)
|
|
{
|
|
MWWorld::Ptr player = MWMechanics::getPlayer();
|
|
MWBase::Environment::get().getMechanicsManager()->itemTaken(player, item, mActor, count, false);
|
|
}
|
|
|
|
return success;
|
|
}
|
|
|
|
bool PickpocketItemModel::stealItem(const MWWorld::Ptr& item, int count)
|
|
{
|
|
MWWorld::Ptr player = MWMechanics::getPlayer();
|
|
MWMechanics::Pickpocket pickpocket(player, mActor);
|
|
if (pickpocket.pick(item, count))
|
|
{
|
|
MWBase::Environment::get().getMechanicsManager()->commitCrime(
|
|
player, mActor, MWBase::MechanicsManager::OT_Pickpocket, ESM::RefId(), 0, true);
|
|
MWBase::Environment::get().getWindowManager()->removeGuiMode(MWGui::GM_Container);
|
|
mPickpocketDetected = true;
|
|
return false;
|
|
}
|
|
else
|
|
player.getClass().skillUsageSucceeded(player, ESM::Skill::Sneak, 1);
|
|
|
|
return true;
|
|
}
|
|
}
|