1
0
Fork 1
mirror of https://github.com/TES3MP/openmw-tes3mp.git synced 2025-01-22 06:53:51 +00:00
openmw-tes3mp/apps/openmw/mwworld/class.cpp

540 lines
15 KiB
C++
Raw Normal View History

#include "class.hpp"
#include <stdexcept>
#include <components/esm/defs.hpp>
2013-08-09 05:34:53 +00:00
#include "../mwbase/environment.hpp"
#include "../mwbase/windowmanager.hpp"
2013-08-11 07:35:19 +00:00
#include "../mwbase/world.hpp"
#include "../mwworld/esmstore.hpp"
2013-08-09 05:34:53 +00:00
#include "ptr.hpp"
#include "refdata.hpp"
2010-08-03 16:20:15 +00:00
#include "nullaction.hpp"
2013-08-09 05:34:53 +00:00
#include "failedaction.hpp"
#include "actiontake.hpp"
#include "containerstore.hpp"
#include "../mwgui/tooltips.hpp"
2013-08-09 05:34:53 +00:00
#include "../mwmechanics/creaturestats.hpp"
2013-08-09 05:34:53 +00:00
#include "../mwmechanics/npcstats.hpp"
namespace MWWorld
{
std::map<std::string, std::shared_ptr<Class> > Class::sClasses;
Class::Class() {}
Class::~Class() {}
void Class::insertObjectRendering (const Ptr& ptr, const std::string& mesh, MWRender::RenderingInterface& renderingInterface) const
{
}
2012-04-08 17:44:11 +00:00
void Class::insertObject(const Ptr& ptr, const std::string& mesh, MWPhysics::PhysicsSystem& physics) const
2012-04-08 17:44:11 +00:00
{
}
bool Class::apply (const MWWorld::Ptr& ptr, const std::string& id, const MWWorld::Ptr& actor) const
{
return false;
}
void Class::skillUsageSucceeded (const MWWorld::Ptr& ptr, int skill, int usageType, float extraFactor) const
{
throw std::runtime_error ("class does not represent an actor");
}
2015-12-18 14:58:23 +00:00
bool Class::canSell (const MWWorld::ConstPtr& item, int npcServices) const
{
return false;
}
2015-12-18 15:44:35 +00:00
int Class::getServices(const ConstPtr &actor) const
2013-05-11 16:38:27 +00:00
{
throw std::runtime_error ("class does not have services");
}
MWMechanics::CreatureStats& Class::getCreatureStats (const Ptr& ptr) const
{
throw std::runtime_error ("class does not have creature stats");
}
2010-08-19 10:49:13 +00:00
MWMechanics::NpcStats& Class::getNpcStats (const Ptr& ptr) const
{
throw std::runtime_error ("class does not have NPC stats");
}
bool Class::hasItemHealth (const ConstPtr& ptr) const
2010-08-03 12:14:04 +00:00
{
return false;
}
int Class::getItemHealth(const ConstPtr &ptr) const
{
if (ptr.getCellRef().getCharge() == -1)
return getItemMaxHealth(ptr);
else
return ptr.getCellRef().getCharge();
}
float Class::getItemNormalizedHealth (const ConstPtr& ptr) const
{
if (getItemMaxHealth(ptr) == 0)
{
return 0.f;
}
else
{
return getItemHealth(ptr) / static_cast<float>(getItemMaxHealth(ptr));
}
}
int Class::getItemMaxHealth (const ConstPtr& ptr) const
2010-08-03 12:14:04 +00:00
{
throw std::runtime_error ("class does not have item health");
}
void Class::hit(const Ptr& ptr, float attackStrength, int type) const
2013-07-24 09:51:42 +00:00
{
throw std::runtime_error("class cannot hit");
2013-07-24 09:51:42 +00:00
}
void Class::block(const Ptr &ptr) const
{
throw std::runtime_error("class cannot block");
}
2016-09-12 15:49:31 +00:00
void Class::onHit(const Ptr& ptr, float damage, bool ishealth, const Ptr& object, const Ptr& attacker, const osg::Vec3f& hitPosition, bool successful) const
{
throw std::runtime_error("class cannot be hit");
}
std::shared_ptr<Action> Class::activate (const Ptr& ptr, const Ptr& actor) const
2010-08-03 16:20:15 +00:00
{
return std::shared_ptr<Action> (new NullAction);
2010-08-03 16:20:15 +00:00
}
std::shared_ptr<Action> Class::use (const Ptr& ptr, bool force) const
2010-08-03 16:20:15 +00:00
{
return std::shared_ptr<Action> (new NullAction);
2010-08-03 16:20:15 +00:00
}
ContainerStore& Class::getContainerStore (const Ptr& ptr) const
2010-08-04 12:37:23 +00:00
{
throw std::runtime_error ("class does not have a container store");
}
InventoryStore& Class::getInventoryStore (const Ptr& ptr) const
{
throw std::runtime_error ("class does not have an inventory store");
}
bool Class::hasInventoryStore(const Ptr &ptr) const
{
return false;
}
2015-12-18 15:50:32 +00:00
bool Class::canLock(const ConstPtr &ptr) const
{
2019-09-11 20:05:24 +00:00
return false;
}
void Class::setRemainingUsageTime (const Ptr& ptr, float duration) const
{
2013-10-16 19:14:35 +00:00
throw std::runtime_error ("class does not support time-based uses");
}
float Class::getRemainingUsageTime (const ConstPtr& ptr) const
{
2013-12-16 12:31:03 +00:00
return -1;
}
2015-12-17 23:12:03 +00:00
std::string Class::getScript (const ConstPtr& ptr) const
{
return "";
2010-08-04 12:37:23 +00:00
}
2011-01-18 09:45:29 +00:00
float Class::getSpeed (const Ptr& ptr) const
{
return 0;
}
float Class::getJump (const Ptr& ptr) const
{
return 0;
}
int Class::getEnchantmentPoints (const MWWorld::ConstPtr& ptr) const
2013-03-16 18:00:14 +00:00
{
throw std::runtime_error ("class does not support enchanting");
}
MWMechanics::Movement& Class::getMovementSettings (const Ptr& ptr) const
{
throw std::runtime_error ("movement settings not supported by class");
}
2015-06-03 17:41:19 +00:00
osg::Vec3f Class::getRotationVector (const Ptr& ptr) const
{
2015-06-03 17:41:19 +00:00
return osg::Vec3f (0, 0, 0);
}
std::pair<std::vector<int>, bool> Class::getEquipmentSlots (const ConstPtr& ptr) const
{
return std::make_pair (std::vector<int>(), false);
}
int Class::getEquipmentSkill (const ConstPtr& ptr) const
{
return -1;
}
int Class::getValue (const ConstPtr& ptr) const
{
throw std::logic_error ("value not supported by this class");
}
2012-05-15 20:31:52 +00:00
float Class::getCapacity (const MWWorld::Ptr& ptr) const
2012-05-15 19:17:00 +00:00
{
2012-05-15 19:34:00 +00:00
throw std::runtime_error ("capacity not supported by this class");
}
2015-12-18 15:00:50 +00:00
float Class::getWeight(const ConstPtr &ptr) const
2013-05-11 16:38:27 +00:00
{
throw std::runtime_error ("weight not supported by this class");
}
2012-05-15 19:34:00 +00:00
float Class::getEncumbrance (const MWWorld::Ptr& ptr) const
{
throw std::runtime_error ("encumbrance not supported by class");
2012-05-15 19:17:00 +00:00
}
2015-12-18 15:39:35 +00:00
bool Class::isEssential (const MWWorld::ConstPtr& ptr) const
2012-10-27 11:33:54 +00:00
{
return false;
}
2013-03-17 21:29:12 +00:00
float Class::getArmorRating (const MWWorld::Ptr& ptr) const
{
throw std::runtime_error("Class does not support armor rating");
}
const Class& Class::get (const std::string& key)
{
if (key.empty())
throw std::logic_error ("Class::get(): attempting to get an empty key");
std::map<std::string, std::shared_ptr<Class> >::const_iterator iter = sClasses.find (key);
if (iter==sClasses.end())
throw std::logic_error ("Class::get(): unknown class key: " + key);
return *iter->second;
}
2015-12-18 15:41:37 +00:00
bool Class::isPersistent(const ConstPtr &ptr) const
2013-05-16 16:50:26 +00:00
{
throw std::runtime_error ("class does not support persistence");
}
void Class::registerClass(const std::string& key, std::shared_ptr<Class> instance)
{
instance->mTypeName = key;
sClasses.insert(std::make_pair(key, instance));
}
2012-03-13 16:05:38 +00:00
2015-12-18 15:09:08 +00:00
std::string Class::getUpSoundId (const ConstPtr& ptr) const
2012-03-13 16:05:38 +00:00
{
throw std::runtime_error ("class does not have an up sound");
}
2015-12-18 15:09:08 +00:00
std::string Class::getDownSoundId (const ConstPtr& ptr) const
2012-03-13 16:05:38 +00:00
{
throw std::runtime_error ("class does not have an down sound");
}
2013-07-18 06:58:21 +00:00
std::string Class::getSoundIdFromSndGen(const Ptr &ptr, const std::string &type) const
{
throw std::runtime_error("class does not support soundgen look up");
}
2015-12-18 14:53:47 +00:00
std::string Class::getInventoryIcon (const MWWorld::ConstPtr& ptr) const
{
throw std::runtime_error ("class does not have any inventory icon");
}
2015-12-19 15:29:07 +00:00
MWGui::ToolTipInfo Class::getToolTipInfo (const ConstPtr& ptr, int count) const
{
throw std::runtime_error ("class does not have a tool tip");
}
bool Class::showsInInventory (const ConstPtr& ptr) const
{
// NOTE: Don't show WerewolfRobe objects in the inventory, or allow them to be taken.
// Vanilla likely uses a hack like this since there's no other way to prevent it from
// being shown or taken.
return (ptr.getCellRef().getRefId() != "werewolfrobe");
}
2015-12-19 15:13:00 +00:00
bool Class::hasToolTip (const ConstPtr& ptr) const
{
return true;
}
2012-05-12 14:17:03 +00:00
2015-12-18 14:56:45 +00:00
std::string Class::getEnchantment (const ConstPtr& ptr) const
2012-05-12 14:17:03 +00:00
{
return "";
}
2012-05-21 08:58:04 +00:00
2015-12-18 15:46:02 +00:00
void Class::adjustScale(const MWWorld::ConstPtr& ptr, osg::Vec3f& scale, bool rendering) const
2012-05-21 08:58:04 +00:00
{
}
2015-12-18 14:51:05 +00:00
std::string Class::getModel(const MWWorld::ConstPtr &ptr) const
{
return "";
}
bool Class::useAnim() const
{
return false;
}
void Class::getModelsToPreload(const Ptr &ptr, std::vector<std::string> &models) const
{
std::string model = getModel(ptr);
if (!model.empty())
models.push_back(model);
}
2015-12-18 15:43:11 +00:00
std::string Class::applyEnchantment(const MWWorld::ConstPtr &ptr, const std::string& enchId, int enchCharge, const std::string& newName) const
2013-03-28 16:41:00 +00:00
{
throw std::runtime_error ("class can't be enchanted");
}
std::pair<int, std::string> Class::canBeEquipped(const MWWorld::ConstPtr &ptr, const MWWorld::Ptr &npc) const
2013-04-05 13:42:05 +00:00
{
return std::make_pair (1, "");
2013-04-05 13:42:05 +00:00
}
void Class::adjustPosition(const MWWorld::Ptr& ptr, bool force) const
{
}
std::shared_ptr<Action> Class::defaultItemActivate(const Ptr &ptr, const Ptr &actor) const
2013-08-09 05:34:53 +00:00
{
if(!MWBase::Environment::get().getWindowManager()->isAllowed(MWGui::GW_Inventory))
return std::shared_ptr<Action>(new NullAction());
2013-08-09 05:34:53 +00:00
if(actor.getClass().isNpc() && actor.getClass().getNpcStats(actor).isWerewolf())
2013-08-09 05:34:53 +00:00
{
2013-08-11 07:35:19 +00:00
const MWWorld::ESMStore &store = MWBase::Environment::get().getWorld()->getStore();
const ESM::Sound *sound = store.get<ESM::Sound>().searchRandom("WolfItem");
std::shared_ptr<MWWorld::Action> action(new MWWorld::FailedAction("#{sWerewolfRefusal}"));
2013-08-11 07:35:19 +00:00
if(sound) action->setSound(sound->mId);
2013-08-09 05:34:53 +00:00
return action;
}
std::shared_ptr<MWWorld::Action> action(new ActionTake(ptr));
2013-08-09 05:34:53 +00:00
action->setSound(getUpSoundId(ptr));
return action;
}
MWWorld::Ptr
2015-12-18 15:24:24 +00:00
Class::copyToCellImpl(const ConstPtr &ptr, CellStore &cell) const
{
2015-12-18 15:24:24 +00:00
throw std::runtime_error("unable to copy class to cell");
}
MWWorld::Ptr
Class::copyToCell(const ConstPtr &ptr, CellStore &cell, int count) const
{
Ptr newPtr = copyToCellImpl(ptr, cell);
newPtr.getCellRef().unsetRefNum(); // This RefNum is only valid within the original cell of the reference
newPtr.getRefData().setCount(count);
return newPtr;
}
MWWorld::Ptr
Class::copyToCell(const ConstPtr &ptr, CellStore &cell, const ESM::Position &pos, int count) const
{
Ptr newPtr = copyToCell(ptr, cell, count);
newPtr.getRefData().setPosition(pos);
return newPtr;
}
2015-12-18 15:15:40 +00:00
bool Class::isBipedal(const ConstPtr &ptr) const
{
return false;
}
bool Class::canFly(const ConstPtr &ptr) const
{
return false;
}
bool Class::canSwim(const ConstPtr &ptr) const
{
return false;
}
bool Class::canWalk(const ConstPtr &ptr) const
{
return false;
}
bool Class::isPureWaterCreature(const ConstPtr& ptr) const
{
2016-11-16 19:15:25 +00:00
return canSwim(ptr)
&& !isBipedal(ptr)
&& !canFly(ptr)
&& !canWalk(ptr);
}
bool Class::isPureFlyingCreature(const ConstPtr& ptr) const
2016-11-16 19:15:25 +00:00
{
return canFly(ptr)
&& !isBipedal(ptr)
&& !canSwim(ptr)
&& !canWalk(ptr);
}
bool Class::isPureLandCreature(const Ptr& ptr) const
{
return canWalk(ptr)
&& !isBipedal(ptr)
&& !canFly(ptr)
&& !canSwim(ptr);
}
bool Class::isMobile(const MWWorld::Ptr& ptr) const
{
return canSwim(ptr) || canWalk(ptr) || canFly(ptr);
}
float Class::getSkill(const MWWorld::Ptr& ptr, int skill) const
{
throw std::runtime_error("class does not support skills");
}
2015-12-18 15:38:14 +00:00
int Class::getBloodTexture (const MWWorld::ConstPtr& ptr) const
2014-01-17 09:52:44 +00:00
{
throw std::runtime_error("class does not support gore");
}
void Class::readAdditionalState (const MWWorld::Ptr& ptr, const ESM::ObjectState& state) const {}
void Class::writeAdditionalState (const MWWorld::ConstPtr& ptr, ESM::ObjectState& state) const {}
2015-12-18 15:28:20 +00:00
int Class::getBaseGold(const MWWorld::ConstPtr& ptr) const
{
throw std::runtime_error("class does not support base gold");
}
2014-04-01 18:15:55 +00:00
2015-12-18 15:29:30 +00:00
bool Class::isClass(const MWWorld::ConstPtr& ptr, const std::string &className) const
2014-04-01 18:15:55 +00:00
{
return false;
}
2019-08-25 13:20:14 +00:00
MWWorld::DoorState Class::getDoorState (const MWWorld::ConstPtr &ptr) const
{
throw std::runtime_error("this is not a door");
}
2019-08-25 13:20:14 +00:00
void Class::setDoorState (const MWWorld::Ptr &ptr, MWWorld::DoorState state) const
{
throw std::runtime_error("this is not a door");
}
float Class::getNormalizedEncumbrance(const Ptr &ptr) const
{
float capacity = getCapacity(ptr);
2018-05-12 17:50:18 +00:00
float encumbrance = getEncumbrance(ptr);
if (encumbrance == 0)
return 0.f;
if (capacity == 0)
return 1.f;
2018-05-12 17:50:18 +00:00
return encumbrance / capacity;
}
2015-12-18 15:11:03 +00:00
std::string Class::getSound(const MWWorld::ConstPtr&) const
{
return std::string();
}
int Class::getBaseFightRating(const ConstPtr &ptr) const
{
throw std::runtime_error("class does not support fight rating");
}
2015-01-27 16:32:21 +00:00
std::string Class::getPrimaryFaction (const MWWorld::ConstPtr& ptr) const
2015-01-27 16:32:21 +00:00
{
return std::string();
}
int Class::getPrimaryFactionRank (const MWWorld::ConstPtr& ptr) const
2015-01-27 16:32:21 +00:00
{
return -1;
}
float Class::getEffectiveArmorRating(const ConstPtr &armor, const Ptr &actor) const
{
throw std::runtime_error("class does not support armor ratings");
}
osg::Vec4f Class::getEnchantmentColor(const MWWorld::ConstPtr& item) const
{
osg::Vec4f result(1,1,1,1);
std::string enchantmentName = item.getClass().getEnchantment(item);
if (enchantmentName.empty())
return result;
const ESM::Enchantment* enchantment = MWBase::Environment::get().getWorld()->getStore().get<ESM::Enchantment>().search(enchantmentName);
if (!enchantment)
return result;
assert (enchantment->mEffects.mList.size());
const ESM::MagicEffect* magicEffect = MWBase::Environment::get().getWorld()->getStore().get<ESM::MagicEffect>().search(
enchantment->mEffects.mList.front().mEffectID);
if (!magicEffect)
return result;
result.x() = magicEffect->mData.mRed / 255.f;
result.y() = magicEffect->mData.mGreen / 255.f;
result.z() = magicEffect->mData.mBlue / 255.f;
return result;
}
void Class::setBaseAISetting(const std::string& id, MWMechanics::CreatureStats::AiSetting setting, int value) const
{
throw std::runtime_error ("class does not have creature stats");
}
float Class::getWalkSpeed(const Ptr& /*ptr*/) const
{
return 0;
}
float Class::getRunSpeed(const Ptr& /*ptr*/) const
{
return 0;
}
float Class::getSwimSpeed(const Ptr& /*ptr*/) const
{
return 0;
}
}