mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-12 03:36:32 +00:00
327 lines
10 KiB
C++
327 lines
10 KiB
C++
|
|
#include "clothing.hpp"
|
|
|
|
#include <components/esm/loadclot.hpp>
|
|
|
|
#include "../mwbase/environment.hpp"
|
|
#include "../mwbase/world.hpp"
|
|
#include "../mwbase/windowmanager.hpp"
|
|
|
|
#include "../mwworld/ptr.hpp"
|
|
#include "../mwworld/actiontake.hpp"
|
|
#include "../mwworld/actionequip.hpp"
|
|
#include "../mwworld/inventorystore.hpp"
|
|
#include "../mwworld/cellstore.hpp"
|
|
#include "../mwworld/physicssystem.hpp"
|
|
#include "../mwworld/nullaction.hpp"
|
|
#include "../mwworld/player.hpp"
|
|
|
|
#include "../mwgui/tooltips.hpp"
|
|
|
|
#include "../mwrender/objects.hpp"
|
|
#include "../mwrender/renderinginterface.hpp"
|
|
|
|
namespace MWClass
|
|
{
|
|
void Clothing::insertObjectRendering (const MWWorld::Ptr& ptr, MWRender::RenderingInterface& renderingInterface) const
|
|
{
|
|
const std::string model = getModel(ptr);
|
|
if (!model.empty()) {
|
|
MWRender::Objects& objects = renderingInterface.getObjects();
|
|
objects.insertBegin(ptr, ptr.getRefData().isEnabled(), false);
|
|
objects.insertMesh(ptr, model);
|
|
}
|
|
}
|
|
|
|
void Clothing::insertObject(const MWWorld::Ptr& ptr, MWWorld::PhysicsSystem& physics) const
|
|
{
|
|
const std::string model = getModel(ptr);
|
|
if(!model.empty())
|
|
physics.addObject(ptr,true);
|
|
}
|
|
|
|
std::string Clothing::getModel(const MWWorld::Ptr &ptr) const
|
|
{
|
|
MWWorld::LiveCellRef<ESM::Clothing> *ref =
|
|
ptr.get<ESM::Clothing>();
|
|
assert(ref->mBase != NULL);
|
|
|
|
const std::string &model = ref->mBase->mModel;
|
|
if (!model.empty()) {
|
|
return "meshes\\" + model;
|
|
}
|
|
return "";
|
|
}
|
|
|
|
std::string Clothing::getName (const MWWorld::Ptr& ptr) const
|
|
{
|
|
MWWorld::LiveCellRef<ESM::Clothing> *ref =
|
|
ptr.get<ESM::Clothing>();
|
|
|
|
return ref->mBase->mName;
|
|
}
|
|
|
|
boost::shared_ptr<MWWorld::Action> Clothing::activate (const MWWorld::Ptr& ptr,
|
|
const MWWorld::Ptr& actor) const
|
|
{
|
|
if (!MWBase::Environment::get().getWindowManager()->isAllowed(MWGui::GW_Inventory))
|
|
return boost::shared_ptr<MWWorld::Action> (new MWWorld::NullAction ());
|
|
|
|
boost::shared_ptr<MWWorld::Action> action(new MWWorld::ActionTake (ptr));
|
|
|
|
action->setSound(getUpSoundId(ptr));
|
|
|
|
return action;
|
|
}
|
|
|
|
std::string Clothing::getScript (const MWWorld::Ptr& ptr) const
|
|
{
|
|
MWWorld::LiveCellRef<ESM::Clothing> *ref =
|
|
ptr.get<ESM::Clothing>();
|
|
|
|
return ref->mBase->mScript;
|
|
}
|
|
|
|
std::pair<std::vector<int>, bool> Clothing::getEquipmentSlots (const MWWorld::Ptr& ptr) const
|
|
{
|
|
MWWorld::LiveCellRef<ESM::Clothing> *ref =
|
|
ptr.get<ESM::Clothing>();
|
|
|
|
std::vector<int> slots;
|
|
|
|
if (ref->mBase->mData.mType==ESM::Clothing::Ring)
|
|
{
|
|
slots.push_back (int (MWWorld::InventoryStore::Slot_LeftRing));
|
|
slots.push_back (int (MWWorld::InventoryStore::Slot_RightRing));
|
|
}
|
|
else
|
|
{
|
|
const int size = 9;
|
|
|
|
static const int sMapping[size][2] =
|
|
{
|
|
{ ESM::Clothing::Shirt, MWWorld::InventoryStore::Slot_Shirt },
|
|
{ ESM::Clothing::Belt, MWWorld::InventoryStore::Slot_Belt },
|
|
{ ESM::Clothing::Robe, MWWorld::InventoryStore::Slot_Robe },
|
|
{ ESM::Clothing::Pants, MWWorld::InventoryStore::Slot_Pants },
|
|
{ ESM::Clothing::Shoes, MWWorld::InventoryStore::Slot_Boots },
|
|
{ ESM::Clothing::LGlove, MWWorld::InventoryStore::Slot_LeftGauntlet },
|
|
{ ESM::Clothing::RGlove, MWWorld::InventoryStore::Slot_RightGauntlet },
|
|
{ ESM::Clothing::Skirt, MWWorld::InventoryStore::Slot_Skirt },
|
|
{ ESM::Clothing::Amulet, MWWorld::InventoryStore::Slot_Amulet }
|
|
};
|
|
|
|
for (int i=0; i<size; ++i)
|
|
if (sMapping[i][0]==ref->mBase->mData.mType)
|
|
{
|
|
slots.push_back (int (sMapping[i][1]));
|
|
break;
|
|
}
|
|
}
|
|
|
|
return std::make_pair (slots, false);
|
|
}
|
|
|
|
int Clothing::getEquipmentSkill (const MWWorld::Ptr& ptr) const
|
|
{
|
|
MWWorld::LiveCellRef<ESM::Clothing> *ref =
|
|
ptr.get<ESM::Clothing>();
|
|
|
|
if (ref->mBase->mData.mType==ESM::Clothing::Shoes)
|
|
return ESM::Skill::Unarmored;
|
|
|
|
return -1;
|
|
}
|
|
|
|
int Clothing::getValue (const MWWorld::Ptr& ptr) const
|
|
{
|
|
MWWorld::LiveCellRef<ESM::Clothing> *ref =
|
|
ptr.get<ESM::Clothing>();
|
|
|
|
return ref->mBase->mData.mValue;
|
|
}
|
|
|
|
void Clothing::registerSelf()
|
|
{
|
|
boost::shared_ptr<Class> instance (new Clothing);
|
|
|
|
registerClass (typeid (ESM::Clothing).name(), instance);
|
|
}
|
|
|
|
std::string Clothing::getUpSoundId (const MWWorld::Ptr& ptr) const
|
|
{
|
|
MWWorld::LiveCellRef<ESM::Clothing> *ref =
|
|
ptr.get<ESM::Clothing>();
|
|
|
|
if (ref->mBase->mData.mType == 8)
|
|
{
|
|
return std::string("Item Ring Up");
|
|
}
|
|
return std::string("Item Clothes Up");
|
|
}
|
|
|
|
std::string Clothing::getDownSoundId (const MWWorld::Ptr& ptr) const
|
|
{
|
|
MWWorld::LiveCellRef<ESM::Clothing> *ref =
|
|
ptr.get<ESM::Clothing>();
|
|
|
|
if (ref->mBase->mData.mType == 8)
|
|
{
|
|
return std::string("Item Ring Down");
|
|
}
|
|
return std::string("Item Clothes Down");
|
|
}
|
|
|
|
std::string Clothing::getInventoryIcon (const MWWorld::Ptr& ptr) const
|
|
{
|
|
MWWorld::LiveCellRef<ESM::Clothing> *ref =
|
|
ptr.get<ESM::Clothing>();
|
|
|
|
return ref->mBase->mIcon;
|
|
}
|
|
|
|
bool Clothing::hasToolTip (const MWWorld::Ptr& ptr) const
|
|
{
|
|
MWWorld::LiveCellRef<ESM::Clothing> *ref =
|
|
ptr.get<ESM::Clothing>();
|
|
|
|
return (ref->mBase->mName != "");
|
|
}
|
|
|
|
MWGui::ToolTipInfo Clothing::getToolTipInfo (const MWWorld::Ptr& ptr) const
|
|
{
|
|
MWWorld::LiveCellRef<ESM::Clothing> *ref =
|
|
ptr.get<ESM::Clothing>();
|
|
|
|
MWGui::ToolTipInfo info;
|
|
info.caption = ref->mBase->mName + MWGui::ToolTips::getCountString(ptr.getRefData().getCount());
|
|
info.icon = ref->mBase->mIcon;
|
|
|
|
std::string text;
|
|
|
|
text += "\n#{sWeight}: " + MWGui::ToolTips::toString(ref->mBase->mData.mWeight);
|
|
text += MWGui::ToolTips::getValueString(ref->mBase->mData.mValue, "#{sValue}");
|
|
|
|
if (MWBase::Environment::get().getWindowManager()->getFullHelp()) {
|
|
text += MWGui::ToolTips::getMiscString(ref->mRef.mOwner, "Owner");
|
|
text += MWGui::ToolTips::getMiscString(ref->mBase->mScript, "Script");
|
|
}
|
|
|
|
info.enchant = ref->mBase->mEnchant;
|
|
if (!info.enchant.empty())
|
|
info.remainingEnchantCharge = ptr.getCellRef().mEnchantmentCharge;
|
|
|
|
info.text = text;
|
|
|
|
return info;
|
|
}
|
|
|
|
std::string Clothing::getEnchantment (const MWWorld::Ptr& ptr) const
|
|
{
|
|
MWWorld::LiveCellRef<ESM::Clothing> *ref =
|
|
ptr.get<ESM::Clothing>();
|
|
|
|
return ref->mBase->mEnchant;
|
|
}
|
|
|
|
void Clothing::applyEnchantment(const MWWorld::Ptr &ptr, const std::string& enchId, int enchCharge, const std::string& newName) const
|
|
{
|
|
MWWorld::LiveCellRef<ESM::Clothing> *ref =
|
|
ptr.get<ESM::Clothing>();
|
|
|
|
ESM::Clothing newItem = *ref->mBase;
|
|
newItem.mId="";
|
|
newItem.mName=newName;
|
|
newItem.mData.mEnchant=enchCharge;
|
|
newItem.mEnchant=enchId;
|
|
const ESM::Clothing *record = MWBase::Environment::get().getWorld()->createRecord (newItem);
|
|
ref->mBase = record;
|
|
}
|
|
|
|
bool Clothing::canEquip(const MWWorld::Ptr &npc, const MWWorld::Ptr &item) const
|
|
{
|
|
MWWorld::InventoryStore& invStore = MWWorld::Class::get(npc).getInventoryStore(npc);
|
|
|
|
// slots that this item can be equipped in
|
|
std::pair<std::vector<int>, bool> slots = MWWorld::Class::get(item).getEquipmentSlots(item);
|
|
|
|
// retrieve ContainerStoreIterator to the item
|
|
MWWorld::ContainerStoreIterator it = invStore.begin();
|
|
for (; it != invStore.end(); ++it)
|
|
{
|
|
if (*it == item)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
assert(it != invStore.end());
|
|
|
|
std::string npcRace = npc.get<ESM::NPC>()->mBase->mRace;
|
|
|
|
for (std::vector<int>::const_iterator slot=slots.first.begin();
|
|
slot!=slots.first.end(); ++slot)
|
|
{
|
|
|
|
// Beast races cannot equip shoes / boots, or full helms (head part vs hair part)
|
|
const ESM::Race* race = MWBase::Environment::get().getWorld()->getStore().get<ESM::Race>().find(npcRace);
|
|
if(race->mData.mFlags & ESM::Race::Beast)
|
|
{
|
|
if (*slot == MWWorld::InventoryStore::Slot_Boots)
|
|
{
|
|
bool allow = true;
|
|
std::vector<ESM::PartReference> parts;
|
|
if(it.getType() == MWWorld::ContainerStore::Type_Clothing)
|
|
parts = it->get<ESM::Clothing>()->mBase->mParts.mParts;
|
|
else
|
|
parts = it->get<ESM::Armor>()->mBase->mParts.mParts;
|
|
for(std::vector<ESM::PartReference>::iterator itr = parts.begin(); itr != parts.end(); ++itr)
|
|
{
|
|
if((*itr).mPart == ESM::PRT_LFoot || (*itr).mPart == ESM::PRT_RFoot)
|
|
{
|
|
allow = false;
|
|
// Only notify the player, not npcs
|
|
if(npc == MWBase::Environment::get().getWorld()->getPlayer().getPlayer() )
|
|
{
|
|
MWBase::Environment::get().getWindowManager()->messageBox ("#{sNotifyMessage15}");
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(!allow)
|
|
return false;
|
|
}
|
|
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
boost::shared_ptr<MWWorld::Action> Clothing::use (const MWWorld::Ptr& ptr) const
|
|
{
|
|
boost::shared_ptr<MWWorld::Action> action(new MWWorld::ActionEquip(ptr));
|
|
|
|
action->setSound(getUpSoundId(ptr));
|
|
|
|
return action;
|
|
}
|
|
|
|
MWWorld::Ptr
|
|
Clothing::copyToCellImpl(const MWWorld::Ptr &ptr, MWWorld::CellStore &cell) const
|
|
{
|
|
MWWorld::LiveCellRef<ESM::Clothing> *ref =
|
|
ptr.get<ESM::Clothing>();
|
|
|
|
return MWWorld::Ptr(&cell.mClothes.insert(*ref), &cell);
|
|
}
|
|
|
|
short Clothing::getEnchantmentPoints (const MWWorld::Ptr& ptr) const
|
|
{
|
|
MWWorld::LiveCellRef<ESM::Clothing> *ref =
|
|
ptr.get<ESM::Clothing>();
|
|
|
|
return ref->mBase->mData.mEnchant;
|
|
}
|
|
}
|